1
0
Fork 0
mirror of https://github.com/TerryCavanagh/VVVVVV.git synced 2024-12-22 17:49:43 +01:00
VVVVVV/desktop_version/src/Game.cpp
Misa 217996b134 Fix UB from out-of-range <stretch>
If there was a scaling mode value (serialized in the XML as <stretch>
for legacy reasons) that was not 0 or 1 or 2, then the rectangle with
the stretch information would not be initialized by get_stretch_info,
which would lead to a crash, either from dividing by zero (most likely)
or from reading an uninitialized value.

To fix this, when reading <stretch>, normalize it to a sane default if
the value is otherwise bogus. And for good measure, an assertion is
added in get_stretch_info() if the value is still somehow bogus.

Fixes #1155.
2024-03-29 20:22:00 -07:00

7898 lines
219 KiB
C++

#define GAME_DEFINITION
#include "Game.h"
#include <sstream>
#include <stdlib.h>
#include <string.h>
#include <tinyxml2.h>
#include "ButtonGlyphs.h"
#include "Constants.h"
#include "CustomLevels.h"
#include "DeferCallbacks.h"
#include "Editor.h"
#include "Entity.h"
#include "Enums.h"
#include "FileSystemUtils.h"
#include "GlitchrunnerMode.h"
#include "Graphics.h"
#include "LevelDebugger.h"
#include "Localization.h"
#include "LocalizationStorage.h"
#include "KeyPoll.h"
#include "MakeAndPlay.h"
#include "Map.h"
#include "Music.h"
#include "Network.h"
#include "RoomnameTranslator.h"
#include "Screen.h"
#include "Script.h"
#include "Unused.h"
#include "UTF8.h"
#include "UtilityClass.h"
#include "VFormat.h"
#include "Vlogging.h"
#include "XMLUtils.h"
static bool GetButtonFromString(const char *pText, SDL_GameControllerButton *button)
{
if (*pText == '0' ||
*pText == 'a' ||
*pText == 'A')
{
*button = SDL_CONTROLLER_BUTTON_A;
return true;
}
if (SDL_strcmp(pText, "1") == 0 ||
*pText == 'b' ||
*pText == 'B')
{
*button = SDL_CONTROLLER_BUTTON_B;
return true;
}
if (*pText == '2' ||
*pText == 'x' ||
*pText == 'X')
{
*button = SDL_CONTROLLER_BUTTON_X;
return true;
}
if (*pText == '3' ||
*pText == 'y' ||
*pText == 'Y')
{
*button = SDL_CONTROLLER_BUTTON_Y;
return true;
}
if (*pText == '4' ||
SDL_strcasecmp(pText, "BACK") == 0)
{
*button = SDL_CONTROLLER_BUTTON_BACK;
return true;
}
if (*pText == '5' ||
SDL_strcasecmp(pText, "GUIDE") == 0)
{
*button = SDL_CONTROLLER_BUTTON_GUIDE;
return true;
}
if (*pText == '6' ||
SDL_strcasecmp(pText, "START") == 0)
{
*button = SDL_CONTROLLER_BUTTON_START;
return true;
}
if (*pText == '7' ||
SDL_strcasecmp(pText, "LS") == 0)
{
*button = SDL_CONTROLLER_BUTTON_LEFTSTICK;
return true;
}
if (*pText == '8' ||
SDL_strcasecmp(pText, "RS") == 0)
{
*button = SDL_CONTROLLER_BUTTON_RIGHTSTICK;
return true;
}
if (*pText == '9' ||
SDL_strcasecmp(pText, "LB") == 0)
{
*button = SDL_CONTROLLER_BUTTON_LEFTSHOULDER;
return true;
}
if (SDL_strcmp(pText, "10") == 0 ||
SDL_strcasecmp(pText, "RB") == 0)
{
*button = SDL_CONTROLLER_BUTTON_RIGHTSHOULDER;
return true;
}
return false;
}
// Unfortunate forward-declare... My hands are pretty tied
static void loadthissummary(
const char* filename,
struct Game::Summary* summary,
tinyxml2::XMLDocument& doc
);
static struct Game::Summary get_summary(
const char* filename,
const char* savename,
tinyxml2::XMLDocument& doc
) {
tinyxml2::XMLHandle hDoc(&doc);
struct Game::Summary summary;
SDL_zero(summary);
if (!FILESYSTEM_loadTiXml2Document(filename, doc))
{
vlog_info("%s not found", savename);
return summary;
}
loadthissummary(savename, &summary, doc);
return summary;
}
void Game::init(void)
{
SDL_strlcpy(magic, "[vVvVvV]game", sizeof(magic));
roomx = 0;
roomy = 0;
prevroomx = 0;
prevroomy = 0;
saverx = 0;
savery = 0;
savecolour = 0;
mutebutton = 0;
muted = false;
musicmuted = false;
musicmutebutton = 0;
glitchrunkludge = false;
gamestate = TITLEMODE;
prevgamestate = TITLEMODE;
hascontrol = true;
jumpheld = false;
advancetext = false;
jumppressed = 0;
gravitycontrol = 0;
teleport = false;
edteleportent = 0; //Added in the port!
companion = 0;
quickrestartkludge = false;
tapleft = 0;
tapright = 0;
press_right = false;
press_left = false;
press_action = false;
press_map = false;
press_interact = false;
interactheld = false;
separate_interact = false;
mapheld = false;
pausescript = false;
completestop = false;
activeactivity = -1;
act_fade = 0;
prev_act_fade = 0;
backgroundtext = false;
startscript = false;
inintermission = false;
alarmon = false;
alarmdelay = 0;
blackout = false;
creditposx = 0;
creditposy = 0;
creditposdelay = 0;
oldcreditposx = 0;
useteleporter = false;
teleport_to_teleporter = 0;
activetele = false;
readytotele = 0;
oldreadytotele = 0;
activity_r = 0;
activity_g = 0;
activity_b = 0;
activity_y = 0;
creditposition = 0;
oldcreditposition = 0;
bestgamedeaths = -1;
//Accessibility Options
colourblindmode = false;
noflashingmode = false;
slowdown = 30;
nodeathmode = false;
nocutscenes = false;
ndmresultcrewrescued = 0;
ndmresulttrinkets = 0;
ndmresulthardestroom.clear();
ndmresulthardestroom_x = hardestroom_x;
ndmresulthardestroom_y = hardestroom_y;
ndmresulthardestroom_specialname = false;
customcol=0;
SDL_memset(crewstats, false, sizeof(crewstats));
SDL_memset(ndmresultcrewstats, false, sizeof(ndmresultcrewstats));
SDL_memset(besttimes, -1, sizeof(besttimes));
SDL_memset(bestframes, -1, sizeof(bestframes));
SDL_memset(besttrinkets, -1, sizeof(besttrinkets));
SDL_memset(bestlives, -1, sizeof(bestlives));
SDL_memset(bestrank, -1, sizeof(bestrank));
crewstats[0] = true;
lastsaved = 0;
//Menu stuff initiliased here:
SDL_memset(unlock, false, sizeof(unlock));
SDL_memset(unlocknotify, false, sizeof(unlock));
currentmenuoption = 0;
menutestmode = false;
current_credits_list_index = 0;
translator_credits_pagenum = 0;
menuxoff = 0;
menuyoff = 0;
menucountdown = 0;
levelpage=0;
playcustomlevel=0;
silence_settings_error = false;
deathcounts = 0;
gameoverdelay = 0;
framecounter = 0;
seed_use_sdl_getticks = false;
editor_disabled = false;
resetgameclock();
gamesaved = false;
gamesavefailed = false;
savetime = "00:00";
savetrinkets = 0;
/* These are only used some of the time. */
saveframes = 0;
saveseconds = 0;
intimetrial = false;
timetrialcountdown = 0;
timetrialshinytarget = 0;
timetrialparlost = false;
timetrialpar = 0;
timetrialcheater = false;
timetrialresulttime = 0;
timetrialresultframes = 0;
timetrialresultshinytarget = 0;
timetrialresulttrinkets = 0;
timetrialresultpar = 0;
timetrialresultdeaths = 0;
start_translator_exploring = false;
translator_exploring = false;
translator_exploring_allowtele = false;
translator_cutscene_test = false;
totalflips = 0;
hardestroom = "Welcome Aboard";
hardestroomdeaths = 0;
hardestroom_x = 13;
hardestroom_y = 5;
hardestroom_specialname = false;
hardestroom_finalstretch = false;
currentroomdeaths=0;
inertia = 1.1f;
swnmode = false;
swntimer = 0;
swngame = SWN_NONE; // Not playing sine wave ninja!
swnstate = 0;
swnstate2 = 0;
swnstate3 = 0;
swnstate4 = 0;
swndelay = 0;
swndeaths = 0;
supercrewmate = false;
scmhurt = false;
scmprogress = 0;
swncolstate = 0;
swncoldelay = 0;
swnrecord = 0;
swnbestrank = 0;
swnrank = 0;
swnmessage = 0;
clearcustomlevelstats();
saveFilePath = FILESYSTEM_getUserSaveDirectory();
tinyxml2::XMLDocument doc;
last_quicksave = get_summary("saves/qsave.vvv", "qsave.vvv", doc);
tinyxml2::XMLDocument docTele;
last_telesave = get_summary("saves/tsave.vvv", "tsave.vvv", doc);
screenshake = flashlight = 0 ;
stat_trinkets = 0;
state = 1;
statedelay = 0;
statelocked = false;
//updatestate();
skipfakeload = false;
ghostsenabled = false;
cliplaytest = false;
playx = 0;
playy = 0;
playrx = 0;
playry = 0;
playgc = 0;
fadetomenu = false;
fadetomenudelay = 0;
fadetolab = false;
fadetolabdelay = 0;
over30mode = true;
showingametimer = false;
ingame_titlemode = false;
ingame_editormode = false;
kludge_ingametemp = Menu::mainmenu;
slidermode = SLIDER_NONE;
disablepause = false;
disableaudiopause = false;
disabletemporaryaudiopause = true;
inputdelay = false;
old_skip_message_timer = 0;
skip_message_timer = 0;
old_mode_indicator_timer = 0;
mode_indicator_timer = 0;
old_screenshot_border_timer = 0;
screenshot_border_timer = 0;
screenshot_saved_success = false;
setdefaultcontrollerbuttons();
}
void Game::setdefaultcontrollerbuttons(void)
{
if (controllerButton_flip.size() < 1)
{
controllerButton_flip.push_back(SDL_CONTROLLER_BUTTON_A);
}
if (controllerButton_map.size() < 1)
{
controllerButton_map.push_back(SDL_CONTROLLER_BUTTON_Y);
}
if (controllerButton_esc.size() < 1)
{
controllerButton_esc.push_back(SDL_CONTROLLER_BUTTON_B);
}
if (controllerButton_restart.size() < 1)
{
controllerButton_restart.push_back(SDL_CONTROLLER_BUTTON_RIGHTSHOULDER);
}
if (controllerButton_interact.size() < 1)
{
controllerButton_interact.push_back(SDL_CONTROLLER_BUTTON_X);
}
/* If one of the arrays was empty, and others weren't, we might now have conflicts...
* A crucial one is if the ACTION button is also "ESC", because then you can't
* fix it with just a controller anymore, which might make the game unplayable.
* This is similar to updatebuttonmappings() in Input.cpp, except less... complete? */
for (size_t f = 0; f < controllerButton_flip.size(); f++)
{
for (size_t e = 0; e < controllerButton_esc.size(); e++)
{
if (controllerButton_flip[f] == controllerButton_esc[e])
{
controllerButton_esc.erase(controllerButton_esc.begin() + e);
break;
}
}
}
}
void Game::lifesequence(void)
{
if (lifeseq > 0)
{
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].invis = false;
if (lifeseq == 2) obj.entities[i].invis = true;
if (lifeseq == 6) obj.entities[i].invis = true;
if (lifeseq >= 8) obj.entities[i].invis = true;
}
if (lifeseq > 5) gravitycontrol = savegc;
lifeseq--;
if (INBOUNDS_VEC(i, obj.entities) && (lifeseq <= 0 || noflashingmode))
{
obj.entities[i].invis = false;
}
}
}
void Game::clearcustomlevelstats(void)
{
//just clearing the array
customlevelstats.clear();
}
void Game::updatecustomlevelstats(std::string clevel, int cscore)
{
if (!map.custommodeforreal)
{
/* We are playtesting, don't update level stats */
return;
}
if (clevel.find("levels/") != std::string::npos)
{
clevel = clevel.substr(7);
}
if (customlevelstats.count(clevel) == 0 || cscore > customlevelstats[clevel])
{
customlevelstats[clevel] = cscore;
}
savecustomlevelstats();
}
void Game::deletecustomlevelstats(void)
{
customlevelstats.clear();
if (!FILESYSTEM_delete("saves/levelstats.vvv"))
{
vlog_error("Error deleting levelstats.vvv");
}
}
#define LOAD_ARRAY_RENAME(ARRAY_NAME, DEST) \
if (SDL_strcmp(pKey, #ARRAY_NAME) == 0 && pText[0] != '\0') \
{ \
/* We're loading in 32-bit integers. If we need more than 16 chars,
* something is seriously wrong */ \
char buffer[16]; \
size_t start = 0; \
size_t i = 0; \
\
while (next_split_s(buffer, sizeof(buffer), &start, pText, ',')) \
{ \
if (i >= SDL_arraysize(DEST)) \
{ \
break; \
} \
\
DEST[i] = help.Int(buffer); \
++i; \
} \
}
#define LOAD_ARRAY(ARRAY_NAME) LOAD_ARRAY_RENAME(ARRAY_NAME, ARRAY_NAME)
void Game::loadcustomlevelstats(void)
{
tinyxml2::XMLDocument doc;
tinyxml2::XMLHandle hDoc(&doc);
if (!FILESYSTEM_loadTiXml2Document("saves/levelstats.vvv", doc))
{
//No levelstats file exists; start new
customlevelstats.clear();
savecustomlevelstats();
return;
}
if (doc.Error())
{
vlog_error("Error parsing levelstats.vvv: %s", doc.ErrorStr());
return;
}
customlevelstats.clear();
tinyxml2::XMLElement* pElem;
tinyxml2::XMLElement* firstElement;
firstElement = hDoc
.FirstChildElement()
.FirstChildElement("Data")
.FirstChildElement()
.ToElement();
// First pass, look for the new system of storing stats
// If they don't exist, then fall back to the old system
for (pElem = firstElement; pElem != NULL; pElem = pElem->NextSiblingElement())
{
const char* pKey = pElem->Value();
const char* pText = pElem->GetText();
if (pText == NULL)
{
pText = "";
}
if (SDL_strcmp(pKey, "stats") == 0)
{
bool file_has_duplicates = false;
for (tinyxml2::XMLElement* stat_el = pElem->FirstChildElement(); stat_el; stat_el = stat_el->NextSiblingElement())
{
int score = 0;
std::string name;
if (stat_el->GetText() != NULL)
{
score = help.Int(stat_el->GetText());
}
if (stat_el->Attribute("name"))
{
name = stat_el->Attribute("name");
}
int existing = customlevelstats.count(name);
if (existing > 0)
{
file_has_duplicates = true;
}
if (existing == 0 || score > customlevelstats[name])
{
customlevelstats[name] = score;
}
}
if (file_has_duplicates)
{
/* This might be really inflated, so simply save the map we have now,
* so we don't have to keep loading a 90 MB file. */
savecustomlevelstats();
}
return;
}
}
// Since we're still here, we must be on the old system
std::vector<std::string> customlevelnames;
std::vector<int> customlevelscores;
for (pElem = firstElement; pElem; pElem=pElem->NextSiblingElement())
{
const char* pKey = pElem->Value();
const char* pText = pElem->GetText() ;
if(pText == NULL)
{
pText = "";
}
if (SDL_strcmp(pKey, "customlevelscore") == 0 && pText[0] != '\0')
{
char buffer[16];
size_t start = 0;
while (next_split_s(buffer, sizeof(buffer), &start, pText, ','))
{
customlevelscores.push_back(help.Int(buffer));
}
}
if (SDL_strcmp(pKey, "customlevelstats") == 0 && pText[0] != '\0')
{
size_t start = 0;
size_t len = 0;
size_t prev_start = 0;
while (next_split(&start, &len, &pText[start], '|'))
{
customlevelnames.push_back(std::string(&pText[prev_start], len));
prev_start = start;
}
}
}
// If the two arrays happen to differ in length, just go with the smallest one
for (size_t i = 0; i < SDL_min(customlevelnames.size(), customlevelscores.size()); i++)
{
const std::string& name = customlevelnames[i];
const int score = customlevelscores[i];
if (customlevelstats.count(name) == 0 || score > customlevelstats[name])
{
customlevelstats[name] = score;
}
}
}
void Game::savecustomlevelstats(void)
{
tinyxml2::XMLDocument doc;
bool already_exists = FILESYSTEM_loadTiXml2Document("saves/levelstats.vvv", doc);
if (!already_exists)
{
vlog_info("No levelstats.vvv found. Creating new file");
}
else if (doc.Error())
{
vlog_error("Error parsing existing levelstats.vvv: %s", doc.ErrorStr());
vlog_info("Creating new levelstats.vvv");
}
xml::update_declaration(doc);
tinyxml2::XMLElement * root = xml::update_element(doc, "Levelstats");
xml::update_comment(root, " Levelstats Save file ");
tinyxml2::XMLElement * msgs = xml::update_element(root, "Data");
int numcustomlevelstats = customlevelstats.size();
if(numcustomlevelstats>=200)numcustomlevelstats=199;
xml::update_tag(msgs, "numcustomlevelstats", numcustomlevelstats);
std::string customlevelscorestr;
std::string customlevelstatsstr;
std::map<std::string, int>::iterator iter;
for (iter = customlevelstats.begin(); iter != customlevelstats.end(); iter++)
{
customlevelscorestr += help.String(iter->second) + ",";
customlevelstatsstr += iter->first + "|";
}
xml::update_tag(msgs, "customlevelscore", customlevelscorestr.c_str());
xml::update_tag(msgs, "customlevelstats", customlevelstatsstr.c_str());
// New system
tinyxml2::XMLElement* msg = xml::update_element_delete_contents(msgs, "stats");
tinyxml2::XMLElement* stat_el;
for (iter = customlevelstats.begin(); iter != customlevelstats.end(); iter++)
{
stat_el = doc.NewElement("stat");
stat_el->SetAttribute("name", iter->first.c_str());
stat_el->LinkEndChild(doc.NewText(help.String(iter->second).c_str()));
msg->LinkEndChild(stat_el);
}
if(FILESYSTEM_saveTiXml2Document("saves/levelstats.vvv", doc))
{
vlog_info("Level stats saved");
}
else
{
vlog_error("Could Not Save level stats!");
vlog_error("Failed: %s%s", saveFilePath, "levelstats.vvv");
}
}
void Game::levelcomplete_textbox(void)
{
graphics.createtextboxflipme("", -1, 12, TEXT_COLOUR("cyan"));
graphics.addline(" ");
graphics.addline("");
graphics.addline("");
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_8X8);
graphics.textboxcenterx();
graphics.setimage(TEXTIMAGE_LEVELCOMPLETE);
graphics.setlinegap(0);
graphics.textboxapplyposition();
}
static void compute_crewmate_textbox(textboxclass* THIS)
{
THIS->lines.clear();
THIS->addline("");
const int extra_cjk_height = (font::height(PR_FONT_INTERFACE) * 4) - 32;
THIS->yp = 64 + 8 + 16 - extra_cjk_height/2;
/* This is a special case for wrapping, we MUST have two lines.
* So just make sure it can't fit in one line. */
const char* text = loc::gettext("You have rescued a crew member!");
std::string wrapped = font::string_wordwrap_balanced(PR_FONT_INTERFACE, text, font::len(PR_FONT_INTERFACE, text)-1);
size_t startline = 0;
size_t newline;
do {
size_t pos_n = wrapped.find('\n', startline);
size_t pos_p = wrapped.find('|', startline);
newline = SDL_min(pos_n, pos_p);
THIS->addline(wrapped.substr(startline, newline-startline));
startline = newline+1;
} while (newline != std::string::npos);
THIS->addline("");
THIS->centertext();
float spaces_per_8 = font::len(PR_FONT_INTERFACE, " ")/8.0f;
THIS->pad(SDL_ceilf(5/spaces_per_8), SDL_ceilf(2/spaces_per_8));
if (!THIS->sprites.empty())
{
THIS->sprites[0].y = 12 + extra_cjk_height/2;
}
}
void Game::crewmate_textbox(const int color)
{
const int extra_cjk_height = (font::height(PR_FONT_INTERFACE) * 4) - 32;
graphics.createtextboxflipme("", -1, 64 + 8 + 16, TEXT_COLOUR("gray"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.addsprite(14, 12 + extra_cjk_height/2, 0, color);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, compute_crewmate_textbox);
graphics.setlinegap(0);
graphics.textboxapplyposition();
}
static void compute_remaining_textbox(textboxclass* THIS)
{
extern Game game;
const int remaining = 6 - game.crewrescued();
char buffer[SCREEN_WIDTH_CHARS + 1];
if (remaining > 0)
{
loc::gettext_plural_fill(buffer, sizeof(buffer), "{n_crew|wordy} remain", "{n_crew|wordy} remains", "n_crew:int", remaining);
}
else
{
SDL_strlcpy(buffer, loc::gettext("All Crew Members Rescued!"), sizeof(buffer));
}
THIS->lines.clear();
THIS->lines.push_back(buffer);
// In CJK, the "You have rescued" box becomes so big we should lower this one a bit...
const int cjk_lowering = font::height(PR_FONT_INTERFACE) - 8;
THIS->yp = 128 + 16 + cjk_lowering;
THIS->pad(2, 2);
}
void Game::remaining_textbox(void)
{
graphics.createtextboxflipme("", -1, 128 + 16, TEXT_COLOUR("gray"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, compute_remaining_textbox);
graphics.textboxapplyposition();
}
static void compute_actionprompt_textbox(textboxclass* THIS)
{
THIS->lines.clear();
char buffer[SCREEN_WIDTH_CHARS + 1];
vformat_buf(
buffer, sizeof(buffer),
loc::gettext("Press {button} to continue"),
"button:but",
vformat_button(ActionSet_InGame, Action_InGame_ACTION)
);
THIS->lines.push_back(buffer);
THIS->pad(1, 1);
}
void Game::actionprompt_textbox(void)
{
graphics.createtextboxflipme("", -1, 196, TEXT_COLOUR("cyan"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, compute_actionprompt_textbox);
graphics.textboxapplyposition();
}
static void savetele_textbox_success(textboxclass* THIS)
{
THIS->lines.clear();
THIS->lines.push_back(loc::gettext("Game Saved"));
THIS->pad(3, 3);
}
static void savetele_textbox_fail(textboxclass* THIS)
{
THIS->lines.clear();
THIS->lines.push_back(loc::gettext("ERROR: Could not save game!"));
THIS->wrap(2);
THIS->pad(1, 1);
}
void Game::savetele_textbox(void)
{
if (inspecial() || map.custommode)
{
return;
}
if (savetele())
{
graphics.createtextboxflipme("", -1, 12, TEXT_COLOUR("gray"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxtimer(25);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, savetele_textbox_success);
}
else
{
graphics.createtextboxflipme("", -1, 12, TEXT_COLOUR("red"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxtimer(50);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, savetele_textbox_fail);
}
graphics.textboxapplyposition();
}
static void wasd_textbox(textboxclass* THIS)
{
THIS->lines.clear();
THIS->lines.push_back(BUTTONGLYPHS_get_wasd_text());
THIS->wrap(4);
THIS->centertext();
THIS->pad(2, 2);
}
static void flip_textbox(textboxclass* THIS)
{
THIS->lines.clear();
char buffer[SCREEN_WIDTH_CHARS*3 + 1];
vformat_buf(
buffer, sizeof(buffer),
loc::gettext("Press {button} to flip"),
"button:but",
vformat_button(ActionSet_InGame, Action_InGame_ACTION)
);
THIS->lines.push_back(buffer);
THIS->wrap(4);
THIS->centertext();
THIS->pad(2, 2);
}
static void arrowkey_textbox(textboxclass* THIS)
{
THIS->lines.clear();
THIS->lines.push_back(loc::gettext("If you prefer, you can press UP or DOWN instead of ACTION to flip."));
THIS->wrap(2);
THIS->centertext();
THIS->pad(1, 1);
}
static void map_textbox(textboxclass* THIS)
{
THIS->lines.clear();
char buffer[SCREEN_WIDTH_CHARS*3 + 1];
vformat_buf(
buffer, sizeof(buffer),
loc::gettext("Press {button} to view map and quicksave"),
"button:but",
vformat_button(ActionSet_InGame, Action_InGame_Map)
);
THIS->lines.push_back(buffer);
THIS->wrap(4);
THIS->centertext();
THIS->pad(2, 2);
}
static void im1_instructions_textbox1(textboxclass* THIS)
{
extern Game game;
THIS->lines.clear();
// Intermission 1 instructional textbox, depends on last saved
const char* floorceiling = graphics.flipmode ? "ceiling" : "floor";
const char* crewmate;
switch (game.lastsaved)
{
case 2:
crewmate = "Vitellary";
break;
case 3:
crewmate = "Vermilion";
break;
case 4:
crewmate = "Verdigris";
break;
case 5:
crewmate = "Victoria";
break;
default:
crewmate = "your companion";
}
char english[SCREEN_WIDTH_TILES*3 + 1]; /* ASCII only */
vformat_buf(english, sizeof(english),
"When you're standing on the {floorceiling}, {crewmate} will try to walk to you.",
"floorceiling:str, crewmate:str",
floorceiling, crewmate
);
THIS->lines.push_back(loc::gettext(english));
THIS->wrap(2);
THIS->padtowidth(36*8);
}
static void im1_instructions_textbox2(textboxclass* THIS)
{
extern Game game;
THIS->lines.clear();
// Intermission 1 instructional textbox, depends on last saved
const char* floorceiling = graphics.flipmode ? "ceiling" : "floor";
const char* crewmate;
switch (game.lastsaved)
{
case 2:
crewmate = "Vitellary";
break;
case 3:
crewmate = "Vermilion";
break;
case 4:
crewmate = "Verdigris";
break;
case 5:
crewmate = "Victoria";
break;
default:
crewmate = "your companion";
}
char english[SCREEN_WIDTH_TILES*3 + 1]; /* ASCII only */
vformat_buf(english, sizeof(english),
"When you're NOT standing on the {floorceiling}, {crewmate} will stop and wait for you.",
"floorceiling:str, crewmate:str",
floorceiling, crewmate
);
THIS->lines.push_back(loc::gettext(english));
THIS->wrap(2);
THIS->padtowidth(36*8);
}
static void im1_instructions_textbox3(textboxclass* THIS)
{
extern Game game;
THIS->lines.clear();
// Intermission 1 instructional textbox, depends on last saved
const char* english;
switch (game.lastsaved)
{
case 2:
case 3:
case 4:
english = "You can't continue to the next room until he is safely across.";
break;
case 5:
english = "You can't continue to the next room until she is safely across.";
break;
default:
english = "You can't continue to the next room until they are safely across.";
}
THIS->lines.push_back(loc::gettext(english));
THIS->wrap(2);
THIS->padtowidth(36*8);
}
/* Also used in foundtrinket() script command. */
void foundtrinket_textbox1(textboxclass* THIS)
{
THIS->lines.clear();
THIS->lines.push_back(loc::gettext("Congratulations!\n\nYou have found a shiny trinket!"));
THIS->wrap(2);
THIS->centertext();
THIS->pad(1, 1);
}
/* Also used in foundtrinket() script command. */
void foundtrinket_textbox2(textboxclass* THIS)
{
extern Game game;
THIS->lines.clear();
const int max_trinkets = map.custommode ? cl.numtrinkets() : 20;
char buffer[SCREEN_WIDTH_CHARS + 1];
vformat_buf(
buffer, sizeof(buffer),
loc::gettext("{n_trinkets|wordy} out of {max_trinkets|wordy}"),
"n_trinkets:int, max_trinkets:int",
game.trinkets(), max_trinkets
);
THIS->lines.push_back(buffer);
if (INBOUNDS_VEC(THIS->other_textbox_index, graphics.textboxes)
&& &graphics.textboxes[THIS->other_textbox_index] != THIS)
{
THIS->yp = 95 + graphics.textboxes[THIS->other_textbox_index].h;
}
THIS->wrap(2);
THIS->centertext();
THIS->pad(1, 1);
}
static void foundcrewmate_textbox1(textboxclass* THIS)
{
THIS->lines.clear();
THIS->lines.push_back(loc::gettext("Congratulations!\n\nYou have found a lost crewmate!"));
THIS->wrap(2);
THIS->centertext();
THIS->pad(1, 1);
}
static void foundcrewmate_textbox2(textboxclass* THIS)
{
extern Game game;
THIS->lines.clear();
const int num_remaining = cl.numcrewmates() - game.crewmates();
if (num_remaining == 0)
{
THIS->lines.push_back(loc::gettext("All crewmates rescued!"));
}
else
{
char buffer[SCREEN_WIDTH_CHARS + 1];
loc::gettext_plural_fill(
buffer, sizeof(buffer),
"{n_crew|wordy} remain", "{n_crew|wordy} remains",
"n_crew:int",
num_remaining
);
THIS->lines.push_back(buffer);
}
if (INBOUNDS_VEC(THIS->other_textbox_index, graphics.textboxes)
&& &graphics.textboxes[THIS->other_textbox_index] != THIS)
{
THIS->yp = 95 + graphics.textboxes[THIS->other_textbox_index].h;
}
THIS->wrap(4);
THIS->centertext();
THIS->pad(2, 2);
}
static void gamecomplete_textbox2(textboxclass* THIS)
{
THIS->lines.clear();
THIS->lines.push_back(loc::gettext("All Crew Members Rescued!"));
}
static void gamecomplete_textbox3(textboxclass* THIS)
{
THIS->lines.clear();
const char* label = loc::gettext("Trinkets Found:");
THIS->lines.push_back(label);
THIS->xp = 170 - font::len(PR_FONT_INTERFACE, label);
}
static void gamecomplete_textbox4(textboxclass* THIS)
{
extern Game game;
THIS->lines.clear();
char buffer[SCREEN_WIDTH_CHARS + 1];
vformat_buf(buffer, sizeof(buffer),
loc::gettext("{gamecomplete_n_trinkets|wordy}"),
"gamecomplete_n_trinkets:int",
game.trinkets()
);
THIS->lines.push_back(buffer);
}
static void gamecomplete_textbox5(textboxclass* THIS)
{
THIS->lines.clear();
const char* label = loc::gettext("Game Time:");
THIS->lines.push_back(label);
THIS->xp = 170 - font::len(PR_FONT_INTERFACE, label);
}
static void gamecomplete_textbox6(textboxclass* THIS)
{
extern Game game;
THIS->lines.clear();
char buffer[SCREEN_WIDTH_CHARS + 1];
help.format_time(buffer, sizeof(buffer), game.saveseconds, game.saveframes, true);
THIS->lines.push_back(buffer);
}
static void gamecomplete_textbox7(textboxclass* THIS)
{
THIS->lines.clear();
const char* label = loc::gettext("Total Flips:");
THIS->lines.push_back(label);
THIS->xp = 170 - font::len(PR_FONT_INTERFACE, label);
}
static void gamecomplete_textbox9(textboxclass* THIS)
{
THIS->lines.clear();
const char* label = loc::gettext("Total Deaths:");
THIS->lines.push_back(label);
THIS->xp = 170 - font::len(PR_FONT_INTERFACE, label);
}
static void gamecomplete_textbox11(textboxclass* THIS)
{
extern Game game;
THIS->lines.clear();
char buffer[SCREEN_WIDTH_CHARS + 1];
loc::gettext_plural_fill(
buffer, sizeof(buffer),
"Hardest Room (with {n_deaths} deaths)",
"Hardest Room (with {n_deaths} death)",
"n_deaths:int",
game.hardestroomdeaths
);
THIS->lines.push_back(buffer);
}
static void gamecomplete_textbox12(textboxclass* THIS)
{
extern Game game;
THIS->lines.clear();
THIS->lines.push_back(
loc::gettext_roomname(
map.custommode,
game.hardestroom_x, game.hardestroom_y,
game.hardestroom.c_str(), game.hardestroom_specialname
)
);
}
void Game::setstate(const int gamestate)
{
if (!statelocked || GlitchrunnerMode_less_than_or_equal(Glitchrunner2_2))
{
state = gamestate;
}
}
void Game::incstate(void)
{
if (!statelocked || GlitchrunnerMode_less_than_or_equal(Glitchrunner2_2))
{
state++;
}
}
void Game::setstatedelay(const int delay)
{
if (!statelocked || GlitchrunnerMode_less_than_or_equal(Glitchrunner2_2))
{
statedelay = delay;
}
}
void Game::lockstate(void)
{
statelocked = true;
}
void Game::unlockstate(void)
{
statelocked = false;
}
void Game::updatestate(void)
{
statedelay--;
if (statedelay <= 0)
{
statedelay = 0;
glitchrunkludge=false;
}
if (statedelay <= 0)
{
switch(state)
{
case 0:
//Do nothing here! Standard game state
if (script.running)
{
if (pausescript && !advancetext)
{
/* Prevent softlocks if we somehow don't have advancetext */
pausescript = false;
}
}
else
{
if (completestop)
{
/* Close potential collection dialogue if warping to ship */
graphics.textboxremove();
graphics.showcutscenebars = false;
}
/* Prevent softlocks if there's no cutscene running right now */
hascontrol = true;
completestop = false;
}
break;
case 1:
//Game initilisation
setstate(0);
break;
case 2:
//Opening cutscene
advancetext = true;
hascontrol = false;
setstate(3);
graphics.createtextbox("To do: write quick", 50, 80, TEXT_COLOUR("cyan"));
graphics.addline("intro to story!");
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_8X8);
//Oh no! what happen to rest of crew etc crash into dimension
break;
case 4:
//End of opening cutscene for now
graphics.createtextbox("", -1, 195, TEXT_COLOUR("gray"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxtimer(60);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, wasd_textbox);
graphics.textboxapplyposition();
setstate(0);
break;
case 5:
//Demo over
advancetext = true;
hascontrol = false;
startscript = true;
newscript="returntohub";
obj.removetrigger(5);
setstate(6);
break;
case 7:
//End of opening cutscene for now
graphics.textboxremove();
hascontrol = true;
advancetext = false;
setstate(0);
break;
case 8:
//Enter dialogue
obj.removetrigger(8);
if (!obj.flags[13])
{
obj.flags[13] = true;
graphics.createtextbox("", -1, 155, TEXT_COLOUR("gray"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxtimer(60);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, map_textbox);
graphics.textboxapplyposition();
}
setstate(0);
break;
case 9:
if (!map.custommode && nocompetitive())
{
returntolab();
startscript = true;
newscript = "disableaccessibility";
setstate(0);
break;
}
//Start SWN Minigame Mode B
obj.removetrigger(9);
swnmode = true;
swngame = SWN_START_SUPERGRAVITRON_STEP_1;
swndelay = 150;
swntimer = 60 * 30;
//set the checkpoint in the middle of the screen
savepoint = 0;
savex = 148;
savey = 100;
savegc = 0;
saverx = roomx;
savery = roomy;
savedir = 0;
setstate(0);
break;
case 10:
//Start SWN Minigame Mode A
obj.removetrigger(10);
swnmode = true;
swngame = SWN_START_GRAVITRON_STEP_1;
swndelay = 150;
swntimer = 60 * 30;
//set the checkpoint in the middle of the screen
savepoint = 0;
savex = 148;
savey = 100;
savegc = 0;
saverx = roomx;
savery = roomy;
savedir = 0;
setstate(0);
break;
case 11:
graphics.textboxremovefast();
graphics.createtextbox("", -1, 3, TEXT_COLOUR("gray"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxtimer(180);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, im1_instructions_textbox2);
graphics.textboxapplyposition();
setstate(0);
break;
case 12:
obj.removetrigger(12);
if (!obj.flags[61])
{
obj.flags[61] = true;
graphics.textboxremovefast();
graphics.createtextbox("", -1, 3, TEXT_COLOUR("gray"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxtimer(120);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, im1_instructions_textbox3);
graphics.textboxapplyposition();
}
setstate(0);
break;
case 13:
//textbox removal
obj.removetrigger(13);
graphics.textboxremovefast();
setstate(0);
break;
case 14:
graphics.createtextbox("", -1, 3, TEXT_COLOUR("gray"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxtimer(280);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, im1_instructions_textbox1);
graphics.textboxapplyposition();
setstate(0);
break;
case 15:
{
//leaving the naughty corner
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[obj.getplayer()].tile = 0;
}
setstate(0);
break;
}
case 16:
{
//entering the naughty corner
int i = obj.getplayer();
if(INBOUNDS_VEC(i, obj.entities) && obj.entities[i].tile == 0)
{
obj.entities[i].tile = 144;
music.playef(Sound_CRY);
}
setstate(0);
break;
}
case 17:
//Arrow key tutorial
obj.removetrigger(17);
if (BUTTONGLYPHS_keyboard_is_active())
{
graphics.createtextbox("", -1, 187, TEXT_COLOUR("gray"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxtimer(100);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, arrowkey_textbox);
graphics.textboxapplyposition();
}
setstate(0);
break;
case 20:
if (!obj.flags[1])
{
obj.flags[1] = true;
setstate(0);
graphics.textboxremove();
}
obj.removetrigger(20);
break;
case 21:
if (!obj.flags[2])
{
obj.flags[2] = true;
setstate(0);
graphics.textboxremove();
}
obj.removetrigger(21);
break;
case 22:
if (!obj.flags[3])
{
graphics.textboxremovefast();
obj.flags[3] = true;
setstate(0);
graphics.createtextbox("", -1, 25, TEXT_COLOUR("gray"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxtimer(60);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, flip_textbox);
graphics.textboxapplyposition();
}
obj.removetrigger(22);
break;
case 30:
//Generic "run script"
if (!obj.flags[4])
{
obj.flags[4] = true;
startscript = true;
newscript="firststeps";
setstate(0);
}
obj.removetrigger(30);
setstate(0);
break;
case 31:
//state = 55; setstatedelay(50);
setstate(0);
setstatedelay(0);
if (!obj.flags[6])
{
obj.flags[6] = true;
obj.flags[5] = true;
startscript = true;
newscript="communicationstation";
setstate(0);
setstatedelay(0);
}
obj.removetrigger(31);
break;
case 32:
//Generic "run script"
if (!obj.flags[7])
{
obj.flags[7] = true;
startscript = true;
newscript="teleporterback";
setstate(0);
}
obj.removetrigger(32);
setstate(0);
break;
case 33:
//Generic "run script"
if (!obj.flags[9])
{
obj.flags[9] = true;
startscript = true;
newscript="rescueblue";
setstate(0);
}
obj.removetrigger(33);
setstate(0);
break;
case 34:
//Generic "run script"
if (!obj.flags[10])
{
obj.flags[10] = true;
startscript = true;
newscript="rescueyellow";
setstate(0);
}
obj.removetrigger(34);
setstate(0);
break;
case 35:
//Generic "run script"
if (!obj.flags[11])
{
obj.flags[11] = true;
startscript = true;
newscript="rescuegreen";
setstate(0);
}
obj.removetrigger(35);
setstate(0);
break;
case 36:
//Generic "run script"
if (!obj.flags[8])
{
obj.flags[8] = true;
startscript = true;
newscript="rescuered";
setstate(0);
}
obj.removetrigger(36);
setstate(0);
break;
case 37:
//Generic "run script"
if (companion == 0)
{
startscript = true;
newscript="int2_yellow";
setstate(0);
}
obj.removetrigger(37);
setstate(0);
break;
case 38:
//Generic "run script"
if (companion == 0)
{
startscript = true;
newscript="int2_red";
setstate(0);
}
obj.removetrigger(38);
setstate(0);
break;
case 39:
//Generic "run script"
if (companion == 0)
{
startscript = true;
newscript="int2_green";
setstate(0);
}
obj.removetrigger(39);
setstate(0);
break;
case 40:
//Generic "run script"
if (companion == 0)
{
startscript = true;
newscript="int2_blue";
setstate(0);
}
obj.removetrigger(40);
setstate(0);
break;
case 41:
//Generic "run script"
if (!obj.flags[60])
{
obj.flags[60] = true;
startscript = true;
if (lastsaved == 2)
{
newscript = "int1yellow_2";
}
else if (lastsaved == 3)
{
newscript = "int1red_2";
}
else if (lastsaved == 4)
{
newscript = "int1green_2";
}
else if (lastsaved == 5)
{
newscript = "int1blue_2";
}
setstate(0);
}
obj.removetrigger(41);
setstate(0);
break;
case 42:
//Generic "run script"
if (!obj.flags[62])
{
obj.flags[62] = true;
startscript = true;
if (lastsaved == 2)
{
newscript = "int1yellow_3";
}
else if (lastsaved == 3)
{
newscript = "int1red_3";
}
else if (lastsaved == 4)
{
newscript = "int1green_3";
}
else if (lastsaved == 5)
{
newscript = "int1blue_3";
}
setstate(0);
}
obj.removetrigger(42);
setstate(0);
break;
case 43:
//Generic "run script"
if (!obj.flags[63])
{
obj.flags[63] = true;
startscript = true;
if (lastsaved == 2)
{
newscript = "int1yellow_4";
}
else if (lastsaved == 3)
{
newscript = "int1red_4";
}
else if (lastsaved == 4)
{
newscript = "int1green_4";
}
else if (lastsaved == 5)
{
newscript = "int1blue_4";
}
setstate(0);
}
obj.removetrigger(43);
setstate(0);
break;
case 44:
//Generic "run script"
if (!obj.flags[64])
{
obj.flags[64] = true;
startscript = true;
if (lastsaved == 2)
{
newscript = "int1yellow_5";
}
else if (lastsaved == 3)
{
newscript = "int1red_5";
}
else if (lastsaved == 4)
{
newscript = "int1green_5";
}
else if (lastsaved == 5)
{
newscript = "int1blue_5";
}
setstate(0);
}
obj.removetrigger(44);
setstate(0);
break;
case 45:
//Generic "run script"
if (!obj.flags[65])
{
obj.flags[65] = true;
startscript = true;
if (lastsaved == 2)
{
newscript = "int1yellow_6";
}
else if (lastsaved == 3)
{
newscript = "int1red_6";
}
else if (lastsaved == 4)
{
newscript = "int1green_6";
}
else if (lastsaved == 5)
{
newscript = "int1blue_6";
}
setstate(0);
}
obj.removetrigger(45);
setstate(0);
break;
case 46:
//Generic "run script"
if (!obj.flags[66])
{
obj.flags[66] = true;
startscript = true;
if (lastsaved == 2)
{
newscript = "int1yellow_7";
}
else if (lastsaved == 3)
{
newscript = "int1red_7";
}
else if (lastsaved == 4)
{
newscript = "int1green_7";
}
else if (lastsaved == 5)
{
newscript = "int1blue_7";
}
setstate(0);
}
obj.removetrigger(46);
setstate(0);
break;
case 47:
//Generic "run script"
if (!obj.flags[69])
{
obj.flags[69] = true;
startscript = true;
newscript="trenchwarfare";
setstate(0);
}
obj.removetrigger(47);
setstate(0);
break;
case 48:
//Generic "run script"
if (!obj.flags[70])
{
obj.flags[70] = true;
startscript = true;
newscript="trinketcollector";
setstate(0);
}
obj.removetrigger(48);
setstate(0);
break;
case 49:
//Start final level music
if (!obj.flags[71])
{
obj.flags[71] = true;
music.niceplay(Music_PREDESTINEDFATEREMIX);
setstate(0);
}
obj.removetrigger(49);
setstate(0);
break;
case 50:
music.playef(Sound_VIOLET);
graphics.createtextbox("", 5, 8, TEXT_COLOUR("purple"));
graphics.textboxcommsrelay("Help! Can anyone hear this message?");
graphics.textboxtimer(60);
incstate();
setstatedelay(100);
break;
case 51:
music.playef(Sound_VIOLET);
graphics.createtextbox("", 5, 8, TEXT_COLOUR("purple"));
graphics.textboxcommsrelay("Verdigris? Are you out there? Are you ok?");
graphics.textboxtimer(60);
incstate();
setstatedelay(100);
break;
case 52:
music.playef(Sound_VIOLET);
graphics.createtextbox("", 5, 8, TEXT_COLOUR("purple"));
graphics.textboxcommsrelay("Please help us! We've crashed and need assistance!");
graphics.textboxtimer(60);
incstate();
setstatedelay(100);
break;
case 53:
music.playef(Sound_VIOLET);
graphics.createtextbox("", 5, 8, TEXT_COLOUR("purple"));
graphics.textboxcommsrelay("Hello? Anyone out there?");
graphics.textboxtimer(60);
incstate();
setstatedelay(100);
break;
case 54:
music.playef(Sound_VIOLET);
graphics.createtextbox("", 5, 8, TEXT_COLOUR("purple"));
graphics.textboxcommsrelay("This is Doctor Violet from the D.S.S. Souleye! Please respond!");
graphics.textboxtimer(60);
incstate();
setstatedelay(100);
break;
case 55:
music.playef(Sound_VIOLET);
graphics.createtextbox("", 5, 8, TEXT_COLOUR("purple"));
graphics.textboxcommsrelay("Please... Anyone...");
graphics.textboxtimer(60);
incstate();
setstatedelay(100);
break;
case 56:
music.playef(Sound_VIOLET);
graphics.createtextbox("", 5, 8, TEXT_COLOUR("purple"));
graphics.textboxcommsrelay("Please be alright, everyone...");
graphics.textboxtimer(60);
setstate(50);
setstatedelay(100);
break;
case 80:
//Used to return to menu from the game
if (graphics.fademode == FADE_FULLY_BLACK)
{
incstate();
}
break;
case 81:
quittomenu();
music.play(Music_PRESENTINGVVVVVV); //should be after quittomenu()
setstate(0);
break;
case 82:
//Time Trial Complete!
obj.removetrigger(82);
if (map.custommode && !map.custommodeforreal)
{
returntoeditor();
ed.show_note(loc::gettext("Time trial completed"));
break;
}
if (translator_exploring)
{
translator_exploring_allowtele = true;
setstate(0);
break;
}
hascontrol = false;
if (timetrialcheater)
{
SDL_zeroa(obj.collect);
}
timetrialresulttime = help.hms_to_seconds(hours, minutes, seconds);
timetrialresultframes = frames;
timetrialresulttrinkets = trinkets();
timetrialresultshinytarget = timetrialshinytarget;
timetrialresultpar = timetrialpar;
timetrialresultdeaths = deathcounts;
timetrialrank = 0;
if (timetrialresulttime <= timetrialpar) timetrialrank++;
if (trinkets() >= timetrialshinytarget) timetrialrank++;
if (deathcounts == 0) timetrialrank++;
if (timetrialresulttime < besttimes[timetriallevel]
|| (timetrialresulttime == besttimes[timetriallevel] && timetrialresultframes < bestframes[timetriallevel])
|| besttimes[timetriallevel]==-1)
{
besttimes[timetriallevel] = timetrialresulttime;
bestframes[timetriallevel] = timetrialresultframes;
}
if (timetrialresulttrinkets > besttrinkets[timetriallevel] || besttrinkets[timetriallevel]==-1)
{
besttrinkets[timetriallevel] = trinkets();
}
if (deathcounts < bestlives[timetriallevel] || bestlives[timetriallevel]==-1)
{
bestlives[timetriallevel] = deathcounts;
}
if (timetrialrank > bestrank[timetriallevel] || bestrank[timetriallevel]==-1)
{
bestrank[timetriallevel] = timetrialrank;
if (timetrialrank >= 3)
{
switch (timetriallevel)
{
case TimeTrial_SPACESTATION1:
unlockAchievement("vvvvvvtimetrial_station1_fixed");
break;
case TimeTrial_LABORATORY:
unlockAchievement("vvvvvvtimetrial_lab_fixed");
break;
case TimeTrial_TOWER:
unlockAchievement("vvvvvvtimetrial_tower_fixed");
break;
case TimeTrial_SPACESTATION2:
unlockAchievement("vvvvvvtimetrial_station2_fixed");
break;
case TimeTrial_WARPZONE:
unlockAchievement("vvvvvvtimetrial_warp_fixed");
break;
case TimeTrial_FINALLEVEL:
unlockAchievement("vvvvvvtimetrial_final_fixed");
}
}
}
savestatsandsettings();
graphics.fademode = FADE_START_FADEOUT;
music.fadeout();
incstate();
break;
case 83:
frames--;
if (graphics.fademode == FADE_FULLY_BLACK)
{
incstate();
}
break;
case 84:
quittomenu();
createmenu(Menu::timetrialcomplete);
setstate(0);
break;
case 85:
//Cutscene skip version of final level change
obj.removetrigger(85);
//Init final stretch
incstate();
music.playef(Sound_FLASH);
music.play(Music_POSITIVEFORCE);
obj.flags[72] = true;
screenshake = 10;
flashlight = 5;
map.finalstretch = true;
map.warpx = false;
map.warpy = false;
map.background = 6;
map.final_colormode = true;
map.final_colorframe = 1;
setstate(0);
break;
//From 90-100 are run scripts for the eurogamer expo only, remove later
case 90:
//Generic "run script"
startscript = true;
newscript="startexpolevel_station1";
obj.removetrigger(90);
setstate(0);
break;
case 91:
//Generic "run script"
startscript = true;
newscript="startexpolevel_lab";
obj.removetrigger(91);
setstate(0);
break;
case 92:
//Generic "run script"
startscript = true;
newscript="startexpolevel_warp";
obj.removetrigger(92);
setstate(0);
break;
case 93:
//Generic "run script"
startscript = true;
newscript="startexpolevel_tower";
obj.removetrigger(93);
setstate(0);
break;
case 94:
//Generic "run script"
startscript = true;
newscript="startexpolevel_station2";
obj.removetrigger(94);
setstate(0);
break;
case 95:
//Generic "run script"
startscript = true;
newscript="startexpolevel_final";
obj.removetrigger(95);
setstate(0);
break;
case 96:
//Used to return to gravitron to game
if (graphics.fademode == FADE_FULLY_BLACK)
{
incstate();
}
break;
case 97:
returntolab();
setstate(0);
break;
case 100:
//
// Meeting crewmate in the warpzone
//
obj.removetrigger(100);
if (!obj.flags[4])
{
obj.flags[4] = true;
incstate();
}
break;
case 101:
{
int i = obj.getplayer();
hascontrol = false;
if (INBOUNDS_VEC(i, obj.entities) && obj.entities[i].onroof > 0 && gravitycontrol == 1)
{
gravitycontrol = 0;
music.playef(Sound_UNFLIP);
}
if (INBOUNDS_VEC(i, obj.entities) && obj.entities[i].onground > 0)
{
incstate();
}
}
break;
case 102:
{
companion = 6;
int i = obj.getcompanion();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].tile = 0;
obj.entities[i].state = 1;
}
advancetext = true;
hascontrol = false;
graphics.createtextbox("Captain! I've been so worried!", 60, 90, 164, 255, 164);
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_8X8);
incstate();
music.playef(Sound_VERDIGRIS);
}
break;
case 104:
graphics.createtextbox("I'm glad you're ok!", 135, 152, TEXT_COLOUR("cyan"));
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_8X8);
incstate();
music.playef(Sound_VIRIDIAN);
graphics.textboxactive();
break;
case 106:
{
graphics.createtextbox("I've been trying to find a", 74, 70, 164, 255, 164);
graphics.addline("way out, but I keep going");
graphics.addline("around in circles...");
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_8X8);
incstate();
music.playef(Sound_CRY);
graphics.textboxactive();
int i = obj.getcompanion();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].tile = 54;
obj.entities[i].state = 0;
}
}
break;
case 108:
graphics.createtextbox("Don't worry! I have a", 125, 152, TEXT_COLOUR("cyan"));
graphics.addline("teleporter key!");
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_8X8);
incstate();
music.playef(Sound_VIRIDIAN);
graphics.textboxactive();
break;
case 110:
{
int i = obj.getcompanion();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].tile = 0;
obj.entities[i].state = 1;
}
graphics.createtextbox("Follow me!", 185, 154, TEXT_COLOUR("cyan"));
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_8X8);
incstate();
music.playef(Sound_VIRIDIAN);
graphics.textboxactive();
}
break;
case 112:
graphics.textboxremove();
hascontrol = true;
advancetext = false;
setstate(0);
break;
case 115:
//
// Test script for space station, totally delete me!
//
hascontrol = false;
incstate();
break;
case 116:
advancetext = true;
hascontrol = false;
graphics.createtextbox("Sorry Eurogamers! Teleporting around", 60 - 20, 200, 255, 64, 64);
graphics.addline("the map doesn't work in this version!");
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_8X8);
graphics.textboxcenterx();
incstate();
break;
case 118:
graphics.textboxremove();
hascontrol = true;
advancetext = false;
setstate(0);
break;
case 120:
//
// Meeting crewmate in the space station
//
obj.removetrigger(120);
if (!obj.flags[5])
{
obj.flags[5] = true;
incstate();
}
break;
case 121:
{
int i = obj.getplayer();
hascontrol = false;
if (INBOUNDS_VEC(i, obj.entities) && obj.entities[i].onground > 0 && gravitycontrol == 0)
{
gravitycontrol = 1;
music.playef(Sound_UNFLIP);
}
if (INBOUNDS_VEC(i, obj.entities) && obj.entities[i].onroof > 0)
{
incstate();
}
}
break;
case 122:
{
companion = 7;
int i = obj.getcompanion();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].tile = 6;
obj.entities[i].state = 1;
}
advancetext = true;
hascontrol = false;
graphics.createtextbox("Captain! You're ok!", 60-10, 90-40, TEXT_COLOUR("yellow"));
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_8X8);
incstate();
music.playef(Sound_VITELLARY);
break;
}
case 124:
{
graphics.createtextbox("I've found a teleporter, but", 60-20, 90 - 40, TEXT_COLOUR("yellow"));
graphics.addline("I can't get it to go anywhere...");
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_8X8);
incstate();
music.playef(Sound_CRY);
graphics.textboxactive();
break;
}
case 126:
graphics.createtextbox("I can help with that!", 125, 152-40, TEXT_COLOUR("cyan"));
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_8X8);
incstate();
music.playef(Sound_VIRIDIAN);
graphics.textboxactive();
break;
case 128:
graphics.createtextbox("I have the teleporter", 130, 152-35, TEXT_COLOUR("cyan"));
graphics.addline("codex for our ship!");
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_8X8);
incstate();
music.playef(Sound_VIRIDIAN);
graphics.textboxactive();
break;
case 130:
{
graphics.createtextbox("Yey! Let's go home!", 60-30, 90-35, TEXT_COLOUR("yellow"));
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_8X8);
incstate();
music.playef(Sound_VITELLARY);
graphics.textboxactive();
int i = obj.getcompanion();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].tile = 6;
obj.entities[i].state = 1;
}
break;
}
case 132:
graphics.textboxremove();
hascontrol = true;
advancetext = false;
setstate(0);
break;
case 200:
//Init final stretch
incstate();
music.playef(Sound_FLASH);
obj.flags[72] = true;
screenshake = 10;
flashlight = 5;
map.finalstretch = true;
map.warpx = false;
map.warpy = false;
map.background = 6;
map.final_colormode = true;
map.final_colorframe = 1;
startscript = true;
newscript="finalterminal_finish";
setstate(0);
break;
// WARNING: If updating this code, make sure to update Map.cpp mapclass::twoframedelayfix()
case 300:
case 301:
case 302:
case 303:
case 304:
case 305:
case 306:
case 307:
case 308:
case 309:
case 310:
case 311:
case 312:
case 313:
case 314:
case 315:
case 316:
case 317:
case 318:
case 319:
case 320:
case 321:
case 322:
case 323:
case 324:
case 325:
case 326:
case 327:
case 328:
case 329:
case 330:
case 331:
case 332:
case 333:
case 334:
case 335:
case 336:
startscript = true;
newscript="custom_"+customscript[state - 300];
obj.removetrigger(state);
setstate(0);
break;
case 1000:
graphics.showcutscenebars = true;
hascontrol = false;
completestop = true;
incstate();
setstatedelay(15);
break;
case 1001:
//Found a trinket!
advancetext = true;
incstate();
graphics.createtextboxflipme("", 50, 85, TEXT_COLOUR("gray"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, foundtrinket_textbox1);
graphics.textboxapplyposition();
graphics.createtextboxflipme("", 50, 95, TEXT_COLOUR("gray"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxindex(graphics.textboxes.size() - 2);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, foundtrinket_textbox2);
graphics.textboxapplyposition();
break;
case 1002:
if (!advancetext)
{
// Prevent softlocks if we somehow don't have advancetext
incstate();
}
break;
case 1003:
graphics.textboxremove();
hascontrol = true;
advancetext = false;
completestop = false;
setstate(0);
if (music.currentsong > -1)
{
music.fadeMusicVolumeIn(3000);
}
graphics.showcutscenebars = false;
break;
case 1010:
graphics.showcutscenebars = true;
hascontrol = false;
completestop = true;
incstate();
setstatedelay(15);
break;
case 1011:
//Found a crewmate!
advancetext = true;
incstate();
graphics.createtextboxflipme("", 50, 85, TEXT_COLOUR("gray"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, foundcrewmate_textbox1);
graphics.textboxapplyposition();
graphics.createtextboxflipme("", 50, 95, TEXT_COLOUR("gray"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxindex(graphics.textboxes.size() - 2);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, foundcrewmate_textbox2);
graphics.textboxapplyposition();
break;
case 1012:
if (!advancetext)
{
// Prevent softlocks if we somehow don't have advancetext
incstate();
}
break;
case 1013:
graphics.textboxremove();
hascontrol = true;
advancetext = false;
completestop = false;
setstate(0);
if(cl.numcrewmates()-crewmates()==0)
{
if(map.custommodeforreal)
{
graphics.fademode = FADE_START_FADEOUT;
setstate(1014);
}
else
{
returntoeditor();
ed.show_note(loc::gettext("Level completed"));
}
}
else
{
if (cl.levmusic > 0)
{
music.fadeMusicVolumeIn(3000);
}
}
graphics.showcutscenebars = false;
break;
case 1014:
frames--;
if (graphics.fademode == FADE_FULLY_BLACK)
{
incstate();
}
break;
case 1015:
//Update level stats
/* FIXME: Have to add check to not save stats for the dumb hack
* `special/stdin.vvvvvv` filename... see elsewhere, grep for
* `special/stdin`! */
if(cl.numcrewmates()-crewmates()==0 && customlevelfilename != "levels/special/stdin.vvvvvv")
{
//Finished level
if (trinkets() >= cl.numtrinkets())
{
//and got all the trinkets!
updatecustomlevelstats(customlevelfilename, 3);
}
else
{
updatecustomlevelstats(customlevelfilename, 1);
}
}
quittomenu();
music.play(Music_PRESENTINGVVVVVV); //should be after quittomenu()
setstate(0);
break;
case 2000:
//Game Saved!
savetele_textbox();
setstate(0);
break;
case 2500:
music.play(Music_PAUSE);
//Activating a teleporter (appear)
incstate();
setstatedelay(15);
flashlight = 5;
screenshake = 90;
music.playef(Sound_FLASH);
break;
case 2501:
//Activating a teleporter 2
incstate();
setstatedelay(0);
flashlight = 5;
screenshake = 0;
//we're done here!
music.playef(Sound_TELEPORT);
break;
case 2502:
{
//Activating a teleporter 2
incstate();
setstatedelay(5);
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].colour = 0;
obj.entities[i].invis = false;
int j = obj.getteleporter();
if (INBOUNDS_VEC(j, obj.entities))
{
obj.entities[i].xp = obj.entities[j].xp+44;
obj.entities[i].yp = obj.entities[j].yp+44;
obj.entities[i].lerpoldxp = obj.entities[i].xp;
obj.entities[i].lerpoldyp = obj.entities[i].yp;
}
obj.entities[i].ay = -6;
obj.entities[i].ax = 6;
obj.entities[i].vy = -6;
obj.entities[i].vx = 6;
}
i = obj.getteleporter();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].tile = 1;
obj.entities[i].colour = 101;
}
break;
}
case 2503:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 10;
}
break;
}
case 2504:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
//obj.entities[i].xp += 10;
}
break;
}
case 2505:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 8;
}
break;
}
case 2506:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 6;
}
break;
}
case 2507:
{
incstate();
break;
}
case 2508:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 2;
}
break;
}
case 2509:
{
incstate();
setstatedelay(15);
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 1;
}
break;
}
case 2510:
advancetext = true;
hascontrol = false;
graphics.createtextbox("Hello?", 125+24, 152-20, TEXT_COLOUR("cyan"));
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_8X8);
incstate();
music.playef(Sound_VIRIDIAN);
graphics.textboxactive();
break;
case 2512:
advancetext = true;
hascontrol = false;
graphics.createtextbox("Is anyone there?", 125+8, 152-24, TEXT_COLOUR("cyan"));
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_8X8);
incstate();
music.playef(Sound_VIRIDIAN);
graphics.textboxactive();
break;
case 2514:
graphics.textboxremove();
hascontrol = true;
advancetext = false;
setstate(0);
music.play(Music_POTENTIALFORANYTHING);
break;
case 3000:
//Activating a teleporter (long version for level complete)
incstate();
setstatedelay(30);
flashlight = 5;
screenshake = 90;
music.playef(Sound_FLASH);
break;
case 3001:
//Activating a teleporter 2
incstate();
setstatedelay(15);
flashlight = 5;
music.playef(Sound_FLASH);
break;
case 3002:
//Activating a teleporter 2
incstate();
setstatedelay(15);
flashlight = 5;
music.playef(Sound_FLASH);
break;
case 3003:
//Activating a teleporter 2
incstate();
setstatedelay(15);
flashlight = 5;
music.playef(Sound_FLASH);
break;
case 3004:
//Activating a teleporter 2
incstate();
setstatedelay(0);
flashlight = 5;
screenshake = 0;
//we're done here!
music.playef(Sound_TELEPORT);
break;
case 3005:
{
//Activating a teleporter 2
incstate();
setstatedelay(50);
switch(companion)
{
case 6:
setstate(3006);
break; //Warp Zone
case 7:
setstate(3020);
break; //Space Station
case 8:
setstate(3040);
break; //Lab
case 9:
setstate(3060);
break; //Tower
case 10:
setstate(3080);
break; //Intermission 2
case 11:
setstate(3085);
break; //Intermission 1
}
if (translator_exploring_allowtele)
{
setstate(3090);
}
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].colour = 0;
obj.entities[i].invis = true;
}
i = obj.getcompanion();
if(INBOUNDS_VEC(i, obj.entities))
{
obj.disableentity(i);
}
i = obj.getteleporter();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].tile = 1;
obj.entities[i].colour = 100;
}
break;
}
case 3006:
//Level complete! (warp zone)
unlocknum(Unlock_WARPZONE_COMPLETE);
lastsaved = 4;
music.play(Music_PATHCOMPLETE);
incstate();
setstatedelay(75);
levelcomplete_textbox();
break;
case 3007:
incstate();
setstatedelay(45);
crewmate_textbox(13);
break;
case 3008:
incstate();
setstatedelay(45);
remaining_textbox();
break;
case 3009:
incstate();
setstatedelay(0);
actionprompt_textbox();
break;
case 3010:
if (jumppressed)
{
incstate();
setstatedelay(30);
graphics.textboxremove();
}
break;
case 3011:
setstate(3070);
setstatedelay(0);
break;
case 3020:
//Level complete! (Space Station 2)
unlocknum(Unlock_SPACESTATION2_COMPLETE);
lastsaved = 2;
music.play(Music_PATHCOMPLETE);
incstate();
setstatedelay(75);
levelcomplete_textbox();
break;
case 3021:
incstate();
setstatedelay(45);
crewmate_textbox(14);
break;
case 3022:
incstate();
setstatedelay(45);
remaining_textbox();
break;
case 3023:
incstate();
setstatedelay(0);
actionprompt_textbox();
break;
case 3024:
if (jumppressed)
{
incstate();
setstatedelay(30);
graphics.textboxremove();
}
break;
case 3025:
setstate(3070);
setstatedelay(0);
break;
case 3040:
//Level complete! (Lab)
unlocknum(Unlock_LABORATORY_COMPLETE);
lastsaved = 5;
music.play(Music_PATHCOMPLETE);
incstate();
setstatedelay(75);
levelcomplete_textbox();
break;
case 3041:
incstate();
setstatedelay(45);
crewmate_textbox(16);
break;
case 3042:
incstate();
setstatedelay(45);
remaining_textbox();
break;
case 3043:
incstate();
setstatedelay(0);
actionprompt_textbox();
break;
case 3044:
if (jumppressed)
{
incstate();
setstatedelay(30);
graphics.textboxremove();
}
break;
case 3045:
setstate(3070);
setstatedelay(0);
break;
case 3050:
//Level complete! (Space Station 1)
unlocknum(Unlock_SPACESTATION1_COMPLETE);
lastsaved = 1;
music.play(Music_PATHCOMPLETE);
incstate();
setstatedelay(75);
levelcomplete_textbox();
break;
case 3051:
incstate();
setstatedelay(45);
crewmate_textbox(20);
break;
case 3052:
incstate();
setstatedelay(45);
remaining_textbox();
break;
case 3053:
incstate();
setstatedelay(0);
actionprompt_textbox();
break;
case 3054:
if (jumppressed)
{
incstate();
setstatedelay(30);
graphics.textboxremove();
teleportscript = "";
}
break;
case 3055:
graphics.fademode = FADE_START_FADEOUT;
incstate();
setstatedelay(10);
break;
case 3056:
if (graphics.fademode == FADE_FULLY_BLACK)
{
startscript = true;
if (crewrescued() == 6)
{
newscript = "startlevel_final";
}
else
{
if (nocutscenes)
{
newscript="bigopenworldskip";
}
else
{
newscript = "bigopenworld";
}
}
setstate(0);
}
break;
case 3060:
//Level complete! (Tower)
unlocknum(Unlock_TOWER_COMPLETE);
lastsaved = 3;
music.play(Music_PATHCOMPLETE);
incstate();
setstatedelay(75);
levelcomplete_textbox();
break;
case 3061:
incstate();
setstatedelay(45);
crewmate_textbox(15);
break;
case 3062:
incstate();
setstatedelay(45);
remaining_textbox();
break;
case 3063:
incstate();
setstatedelay(0);
actionprompt_textbox();
break;
case 3064:
if (jumppressed)
{
incstate();
setstatedelay(30);
graphics.textboxremove();
}
break;
case 3065:
setstate(3070);
setstatedelay(0);
break;
case 3070:
graphics.fademode = FADE_START_FADEOUT;
incstate();
break;
case 3071:
if (graphics.fademode == FADE_FULLY_BLACK)
{
incstate();
}
break;
case 3072:
//Ok, we need to adjust some flags based on who've we've rescued. Some of there conversation options
//change depending on when they get back to the ship.
if (lastsaved == 2)
{
if (crewstats[3]) obj.flags[25] = true;
if (crewstats[4]) obj.flags[26] = true;
if (crewstats[5]) obj.flags[24] = true;
}
else if (lastsaved == 3)
{
if (crewstats[2]) obj.flags[50] = true;
if (crewstats[4]) obj.flags[49] = true;
if (crewstats[5]) obj.flags[48] = true;
}
else if (lastsaved == 4)
{
if (crewstats[2]) obj.flags[54] = true;
if (crewstats[3]) obj.flags[55] = true;
if (crewstats[5]) obj.flags[56] = true;
}
else if (lastsaved == 5)
{
if (crewstats[2]) obj.flags[37] = true;
if (crewstats[3]) obj.flags[38] = true;
if (crewstats[4]) obj.flags[39] = true;
}
//We're pitch black now, make a decision
companion = 0;
if (crewrescued() == 6)
{
startscript = true;
newscript="startlevel_final";
setstate(0);
}
else if (crewrescued() == 4)
{
companion = 11;
supercrewmate = true;
scmprogress = 0;
startscript = true;
newscript = "intermission_1";
obj.flags[19] = true;
if (lastsaved == 2) obj.flags[32] = true;
if (lastsaved == 3) obj.flags[35] = true;
if (lastsaved == 4) obj.flags[34] = true;
if (lastsaved == 5) obj.flags[33] = true;
setstate(0);
}
else if (crewrescued() == 5)
{
startscript = true;
newscript = "intermission_2";
obj.flags[20] = true;
if (lastsaved == 2) obj.flags[32] = true;
if (lastsaved == 3) obj.flags[35] = true;
if (lastsaved == 4) obj.flags[34] = true;
if (lastsaved == 5) obj.flags[33] = true;
setstate(0);
}
else
{
startscript = true;
newscript="regularreturn";
setstate(0);
}
break;
case 3080:
//returning from an intermission, very like 3070
if (inintermission)
{
graphics.fademode = FADE_START_FADEOUT;
companion = 0;
setstate(3100);
}
else
{
unlocknum(Unlock_INTERMISSION2_COMPLETE);
graphics.fademode = FADE_START_FADEOUT;
companion = 0;
incstate();
}
break;
case 3081:
if (graphics.fademode == FADE_FULLY_BLACK)
{
incstate();
}
break;
case 3082:
map.finalmode = false;
startscript = true;
newscript="regularreturn";
setstate(0);
break;
case 3085:
//returning from an intermission, very like 3070
//return to menu from here
if (inintermission)
{
companion = 0;
supercrewmate = false;
incstate();
graphics.fademode = FADE_START_FADEOUT;
music.fadeout();
setstate(3100);
}
else
{
unlocknum(Unlock_INTERMISSION1_COMPLETE);
graphics.fademode = FADE_START_FADEOUT;
companion = 0;
supercrewmate = false;
incstate();
}
break;
case 3086:
if (graphics.fademode == FADE_FULLY_BLACK)
{
incstate();
}
break;
case 3087:
map.finalmode = false;
startscript = true;
newscript="regularreturn";
setstate(0);
break;
case 3090:
/* Teleporting in translator_exploring should be just like
* the intermission replays: simply return to the menu */
companion = 0;
supercrewmate = false;
graphics.fademode = FADE_START_FADEOUT;
music.fadeout();
setstate(3100);
break;
case 3091:
/* Different Final Level ending for translator_exploring */
music.fadeout();
incstate();
setstatedelay(60);
break;
case 3092:
graphics.fademode = FADE_START_FADEOUT;
setstate(3100);
break;
case 3100:
if (graphics.fademode == FADE_FULLY_BLACK)
{
incstate();
}
break;
case 3101:
quittomenu();
music.play(Music_PRESENTINGVVVVVV); //should be after quittomenu();
setstate(0);
break;
case 3500:
music.fadeout();
incstate();
setstatedelay(120);
break;
case 3501:
//Game complete!
unlockAchievement("vvvvvvgamecomplete");
unlocknum(UnlockTrophy_GAME_COMPLETE);
crewstats[0] = true;
incstate();
setstatedelay(75);
music.play(Music_PLENARY);
graphics.createtextboxflipme("", -1, 12, TEXT_COLOUR("cyan"));
graphics.addline(" ");
graphics.addline("");
graphics.addline("");
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_8X8);
graphics.textboxcenterx();
graphics.setimage(TEXTIMAGE_GAMECOMPLETE);
graphics.setlinegap(0);
graphics.textboxapplyposition();
break;
case 3502:
incstate();
setstatedelay(45+15);
graphics.createtextboxflipme("", -1, 64, TEXT_COLOUR("transparent"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, gamecomplete_textbox2);
graphics.textboxapplyposition();
saveframes = frames;
saveseconds = help.hms_to_seconds(hours, minutes, seconds);
break;
case 3503:
incstate();
setstatedelay(45);
graphics.createtextboxflipme("", 170, 84, TEXT_COLOUR("transparent"));
graphics.textboxprintflags(PR_FONT_INTERFACE | PR_RTL_XFLIP);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, gamecomplete_textbox3);
graphics.textboxapplyposition();
graphics.createtextboxflipme("", 180, 84, TEXT_COLOUR("transparent"));
graphics.textboxprintflags(PR_FONT_INTERFACE | PR_RTL_XFLIP);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, gamecomplete_textbox4);
graphics.textboxapplyposition();
break;
case 3504:
incstate();
setstatedelay(45+15);
graphics.createtextboxflipme("", 170, 96, TEXT_COLOUR("transparent"));
graphics.textboxprintflags(PR_FONT_INTERFACE | PR_RTL_XFLIP);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, gamecomplete_textbox5);
graphics.textboxapplyposition();
graphics.createtextboxflipme("", 180, 96, TEXT_COLOUR("transparent"));
graphics.textboxprintflags(PR_FONT_INTERFACE | PR_RTL_XFLIP);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, gamecomplete_textbox6);
graphics.textboxapplyposition();
break;
case 3505:
incstate();
setstatedelay(45);
graphics.createtextboxflipme("", 170, 123, TEXT_COLOUR("transparent"));
graphics.textboxprintflags(PR_FONT_INTERFACE | PR_RTL_XFLIP);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, gamecomplete_textbox7);
graphics.textboxapplyposition();
graphics.createtextboxflipme(help.String(totalflips), 180, 123, TEXT_COLOUR("transparent"));
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_INTERFACE | PR_RTL_XFLIP);
break;
case 3506:
incstate();
setstatedelay(45+15);
graphics.createtextboxflipme("", 170, 135, TEXT_COLOUR("transparent"));
graphics.textboxprintflags(PR_FONT_INTERFACE | PR_RTL_XFLIP);
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, gamecomplete_textbox9);
graphics.textboxapplyposition();
graphics.createtextboxflipme(help.String(deathcounts), 180, 135, TEXT_COLOUR("transparent"));
graphics.textboxoriginalcontextauto();
graphics.textboxprintflags(PR_FONT_INTERFACE | PR_RTL_XFLIP);
break;
case 3507:
incstate();
setstatedelay(45+15);
graphics.createtextboxflipme("", -1, 158, TEXT_COLOUR("transparent"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, gamecomplete_textbox11);
graphics.textboxapplyposition();
graphics.createtextboxflipme("", -1, 170, TEXT_COLOUR("transparent"));
graphics.textboxprintflags(PR_FONT_INTERFACE);
graphics.textboxcenterx();
graphics.textboxtranslate(TEXTTRANSLATE_FUNCTION, gamecomplete_textbox12);
graphics.textboxapplyposition();
break;
case 3508:
incstate();
setstatedelay(0);
actionprompt_textbox();
break;
case 3509:
if (jumppressed)
{
incstate();
setstatedelay(30);
graphics.textboxremove();
}
break;
case 3510:
//Save stats and stuff here
if (!obj.flags[73])
{
//flip mode complete
unlockAchievement("vvvvvvgamecompleteflip");
unlocknum(UnlockTrophy_FLIPMODE_COMPLETE);
}
#ifndef MAKEANDPLAY
if (!map.custommode)
{
if (bestgamedeaths == -1)
{
bestgamedeaths = deathcounts;
}
else
{
if (deathcounts < bestgamedeaths)
{
bestgamedeaths = deathcounts;
}
}
}
#endif
if (bestgamedeaths > -1) {
if (bestgamedeaths <= 500) {
unlockAchievement("vvvvvvcomplete500");
}
if (bestgamedeaths <= 250) {
unlockAchievement("vvvvvvcomplete250");
}
if (bestgamedeaths <= 100) {
unlockAchievement("vvvvvvcomplete100");
}
if (bestgamedeaths <= 50) {
unlockAchievement("vvvvvvcomplete50");
}
}
if (nodeathmode)
{
unlockAchievement("vvvvvvmaster"); //bloody hell
unlocknum(UnlockTrophy_NODEATHMODE_COMPLETE);
setstate(3520);
setstatedelay(0);
}
else
{
setstatedelay(120);
incstate();
}
savestatsandsettings();
break;
case 3511:
{
//Activating a teleporter (long version for level complete)
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].colour = 102;
}
incstate();
setstatedelay(30);
flashlight = 5;
screenshake = 90;
music.playef(Sound_FLASH);
break;
}
case 3512:
//Activating a teleporter 2
incstate();
setstatedelay(15);
flashlight = 5;
music.playef(Sound_FLASH);
break;
case 3513:
//Activating a teleporter 2
incstate();
setstatedelay(15);
flashlight = 5;
music.playef(Sound_FLASH);
break;
case 3514:
//Activating a teleporter 2
incstate();
setstatedelay(15);
flashlight = 5;
music.playef(Sound_FLASH);
break;
case 3515:
{
//Activating a teleporter 2
incstate();
setstatedelay(0);
flashlight = 5;
screenshake = 0;
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].colour = 0;
obj.entities[i].invis = true;
}
//we're done here!
music.playef(Sound_TELEPORT);
setstatedelay(60);
break;
}
case 3516:
graphics.fademode = FADE_START_FADEOUT;
incstate();
break;
case 3517:
if (graphics.fademode == FADE_FULLY_BLACK)
{
incstate();
setstatedelay(30);
}
break;
case 3518:
graphics.fademode = FADE_START_FADEIN;
setstate(0);
setstatedelay(30);
map.finalmode = false;
map.final_colormode = false;
map.final_mapcol = 0;
map.final_colorframe = 0;
map.finalstretch = false;
obj.flags[72] = false;
graphics.setbars(320);
teleport_to_new_area = true;
teleportscript = "gamecomplete";
break;
case 3520:
//NO DEATH MODE COMPLETE JESUS
hascontrol = false;
crewstats[0] = true;
graphics.fademode = FADE_START_FADEOUT;
incstate();
break;
case 3521:
if (graphics.fademode == FADE_FULLY_BLACK)
{
incstate();
}
break;
case 3522:
copyndmresults();
quittomenu();
createmenu(Menu::nodeathmodecomplete);
setstate(0);
break;
case 4000:
//Activating a teleporter (short version)
state++; // Increment manually -- gamestate modification might be locked at this point
statedelay = 10;
flashlight = 5;
screenshake = 10;
music.playef(Sound_FLASH);
break;
case 4001:
//Activating a teleporter 2
state++;
statedelay = 0;
flashlight = 5;
screenshake = 0;
//we're done here!
music.playef(Sound_TELEPORT);
break;
case 4002:
{
//Activating a teleporter 2
state++;
statedelay = 10;
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].colour = 0;
obj.entities[i].invis = true;
}
i = obj.getteleporter();
if(INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].tile = 1;
obj.entities[i].colour = 100;
}
break;
}
case 4003:
state = 0;
statedelay = 0;
teleport_to_new_area = true;
unlockstate();
break;
case 4010:
//Activating a teleporter (default appear)
incstate();
setstatedelay(15);
flashlight = 5;
screenshake = 90;
music.playef(Sound_FLASH);
break;
case 4011:
//Activating a teleporter 2
incstate();
setstatedelay(0);
flashlight = 5;
screenshake = 0;
music.playef(Sound_TELEPORT);
break;
case 4012:
{
//Activating a teleporter 2
incstate();
setstatedelay(5);
int i = obj.getplayer();
int j = obj.getteleporter();
if (INBOUNDS_VEC(i, obj.entities))
{
if (INBOUNDS_VEC(j, obj.entities))
{
obj.entities[i].xp = obj.entities[j].xp+44;
obj.entities[i].yp = obj.entities[j].yp+44;
obj.entities[i].lerpoldxp = obj.entities[i].xp;
obj.entities[i].lerpoldyp = obj.entities[i].yp;
obj.entities[j].tile = 2;
obj.entities[j].colour = 101;
}
obj.entities[i].colour = 0;
obj.entities[i].invis = false;
obj.entities[i].dir = 1;
obj.entities[i].ay = -6;
obj.entities[i].ax = 6;
obj.entities[i].vy = -6;
obj.entities[i].vx = 6;
}
break;
}
case 4013:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 10;
}
break;
}
case 4014:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 10;
}
break;
}
case 4015:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 8;
}
break;
}
case 4016:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 6;
}
break;
}
case 4017:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 3;
}
break;
}
case 4018:
{
incstate();
setstatedelay(15);
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 1;
}
break;
}
case 4019:
{
if (intimetrial || nodeathmode || inintermission)
{
}
else
{
savetele();
}
int i = obj.getteleporter();
activetele = true;
if (INBOUNDS_VEC(i, obj.entities))
{
teleblock.x = obj.entities[i].xp - 32;
teleblock.y = obj.entities[i].yp - 32;
}
teleblock.w = 160;
teleblock.h = 160;
hascontrol = true;
advancetext = false;
setstate(0);
break;
}
case 4020:
//Activating a teleporter (default appear)
incstate();
setstatedelay(15);
flashlight = 5;
screenshake = 90;
music.playef(Sound_FLASH);
break;
case 4021:
//Activating a teleporter 2
incstate();
setstatedelay(0);
flashlight = 5;
screenshake = 0;
music.playef(Sound_TELEPORT);
break;
case 4022:
{
//Activating a teleporter 2
incstate();
setstatedelay(5);
int i = obj.getplayer();
int j = obj.getteleporter();
if (INBOUNDS_VEC(i, obj.entities))
{
if (INBOUNDS_VEC(j, obj.entities))
{
obj.entities[i].xp = obj.entities[j].xp+44;
obj.entities[i].yp = obj.entities[j].yp+44;
obj.entities[i].lerpoldxp = obj.entities[i].xp;
obj.entities[i].lerpoldyp = obj.entities[i].yp;
obj.entities[j].tile = 2;
obj.entities[j].colour = 101;
}
obj.entities[i].colour = 0;
obj.entities[i].invis = false;
obj.entities[i].dir = 1;
obj.entities[i].ay = -6;
obj.entities[i].ax = 6;
obj.entities[i].vy = -6;
obj.entities[i].vx = 6;
}
break;
}
case 4023:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 12;
}
break;
}
case 4024:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 12;
}
break;
}
case 4025:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 10;
}
break;
}
case 4026:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 8;
}
break;
}
case 4027:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 5;
}
break;
}
case 4028:
{
incstate();
setstatedelay(15);
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 2;
}
break;
}
case 4029:
hascontrol = true;
advancetext = false;
setstate(0);
break;
case 4030:
//Activating a teleporter (default appear)
incstate();
setstatedelay(15);
flashlight = 5;
screenshake = 90;
music.playef(Sound_FLASH);
break;
case 4031:
//Activating a teleporter 2
incstate();
setstatedelay(0);
flashlight = 5;
screenshake = 0;
music.playef(Sound_TELEPORT);
break;
case 4032:
{
//Activating a teleporter 2
incstate();
setstatedelay(5);
int i = obj.getplayer();
int j = obj.getteleporter();
if (INBOUNDS_VEC(i, obj.entities))
{
if (INBOUNDS_VEC(j, obj.entities))
{
obj.entities[i].xp = obj.entities[j].xp+44;
obj.entities[i].yp = obj.entities[j].yp+44;
obj.entities[i].lerpoldxp = obj.entities[i].xp;
obj.entities[i].lerpoldyp = obj.entities[i].yp;
obj.entities[j].tile = 2;
obj.entities[j].colour = 101;
}
obj.entities[i].colour = 0;
obj.entities[i].invis = false;
obj.entities[i].dir = 0;
obj.entities[i].ay = -6;
obj.entities[i].ax = -6;
obj.entities[i].vy = -6;
obj.entities[i].vx = -6;
}
break;
}
case 4033:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp -= 12;
}
break;
}
case 4034:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp -= 12;
}
break;
}
case 4035:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp -= 10;
}
break;
}
case 4036:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp -= 8;
}
break;
}
case 4037:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp -= 5;
}
break;
}
case 4038:
{
incstate();
setstatedelay(15);
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp -= 2;
}
break;
}
case 4039:
hascontrol = true;
advancetext = false;
setstate(0);
break;
case 4040:
//Activating a teleporter (default appear)
incstate();
setstatedelay(15);
flashlight = 5;
screenshake = 90;
music.playef(Sound_FLASH);
break;
case 4041:
//Activating a teleporter 2
incstate();
setstatedelay(0);
flashlight = 5;
screenshake = 0;
music.playef(Sound_TELEPORT);
break;
case 4042:
{
//Activating a teleporter 2
incstate();
setstatedelay(5);
int i = obj.getplayer();
int j = obj.getteleporter();
if (INBOUNDS_VEC(i, obj.entities))
{
if (INBOUNDS_VEC(j, obj.entities))
{
obj.entities[i].xp = obj.entities[j].xp+44;
obj.entities[i].yp = obj.entities[j].yp+44;
obj.entities[i].lerpoldxp = obj.entities[i].xp;
obj.entities[i].lerpoldyp = obj.entities[i].yp;
obj.entities[j].tile = 2;
obj.entities[j].colour = 101;
}
obj.entities[i].colour = 0;
obj.entities[i].invis = false;
obj.entities[i].dir = 1;
obj.entities[i].ay = -6;
obj.entities[i].ax = 6;
obj.entities[i].vy = -6;
obj.entities[i].vx = 6;
}
break;
}
case 4043:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 12;
obj.entities[i].yp -= 15;
}
break;
}
case 4044:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 12;
obj.entities[i].yp -= 10;
}
break;
}
case 4045:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 12;
obj.entities[i].yp -= 10;
}
break;
}
case 4046:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 8;
obj.entities[i].yp -= 8;
}
break;
}
case 4047:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 6;
obj.entities[i].yp -= 8;
}
break;
}
case 4048:
{
incstate();
setstatedelay(15);
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 3;
}
break;
}
case 4049:
hascontrol = true;
advancetext = false;
setstate(0);
break;
case 4050:
//Activating a teleporter (default appear)
incstate();
setstatedelay(15);
flashlight = 5;
screenshake = 90;
music.playef(Sound_FLASH);
break;
case 4051:
//Activating a teleporter 2
incstate();
setstatedelay(0);
flashlight = 5;
screenshake = 0;
music.playef(Sound_TELEPORT);
break;
case 4052:
{
//Activating a teleporter 2
incstate();
setstatedelay(5);
int i = obj.getplayer();
int j = obj.getteleporter();
if (INBOUNDS_VEC(i, obj.entities))
{
if (INBOUNDS_VEC(j, obj.entities))
{
obj.entities[i].xp = obj.entities[j].xp+44;
obj.entities[i].yp = obj.entities[j].yp+44;
obj.entities[i].lerpoldxp = obj.entities[i].xp;
obj.entities[i].lerpoldyp = obj.entities[i].yp;
obj.entities[j].tile = 2;
obj.entities[j].colour = 101;
}
obj.entities[i].colour = 0;
obj.entities[i].invis = false;
obj.entities[i].dir = 1;
obj.entities[i].ay = -6;
obj.entities[i].ax = 6;
obj.entities[i].vy = -6;
obj.entities[i].vx = 6;
}
break;
}
case 4053:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 4;
obj.entities[i].yp -= 15;
}
break;
}
case 4054:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 4;
obj.entities[i].yp -= 10;
}
break;
}
case 4055:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 4;
obj.entities[i].yp -= 10;
}
break;
}
case 4056:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 4;
obj.entities[i].yp -= 8;
}
break;
}
case 4057:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 2;
obj.entities[i].yp -= 8;
}
break;
}
case 4058:
{
incstate();
setstatedelay(15);
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 1;
}
break;
}
case 4059:
hascontrol = true;
advancetext = false;
setstate(0);
break;
case 4060:
//Activating a teleporter (default appear)
incstate();
setstatedelay(15);
flashlight = 5;
screenshake = 90;
music.playef(Sound_FLASH);
break;
case 4061:
//Activating a teleporter 2
incstate();
setstatedelay(0);
flashlight = 5;
screenshake = 0;
music.playef(Sound_TELEPORT);
break;
case 4062:
{
//Activating a teleporter 2
incstate();
setstatedelay(5);
int i = obj.getplayer();
int j = obj.getteleporter();
if (INBOUNDS_VEC(i, obj.entities))
{
if (INBOUNDS_VEC(j, obj.entities))
{
obj.entities[i].xp = obj.entities[j].xp+44;
obj.entities[i].yp = obj.entities[j].yp+44;
obj.entities[i].lerpoldxp = obj.entities[i].xp;
obj.entities[i].lerpoldyp = obj.entities[i].yp;
obj.entities[j].tile = 2;
obj.entities[j].colour = 101;
}
obj.entities[i].colour = 0;
obj.entities[i].invis = false;
obj.entities[i].dir = 0;
obj.entities[i].ay = -6;
obj.entities[i].ax = -6;
obj.entities[i].vy = -6;
obj.entities[i].vx = -6;
}
break;
}
case 4063:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp -= 28;
obj.entities[i].yp -= 8;
}
break;
}
case 4064:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp -= 28;
obj.entities[i].yp -= 8;
}
break;
}
case 4065:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp -= 25;
}
break;
}
case 4066:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp -= 25;
}
break;
}
case 4067:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp -= 20;
}
break;
}
case 4068:
{
incstate();
setstatedelay(15);
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp -= 16;
}
break;
}
case 4069:
hascontrol = true;
advancetext = false;
setstate(0);
break;
case 4070:
//Activating a teleporter (special for final script, player has colour changed to match rescued crewmate)
incstate();
setstatedelay(15);
flashlight = 5;
screenshake = 90;
music.playef(Sound_FLASH);
break;
case 4071:
//Activating a teleporter 2
incstate();
setstatedelay(0);
flashlight = 5;
screenshake = 0;
music.playef(Sound_TELEPORT);
break;
case 4072:
{
//Activating a teleporter 2
incstate();
setstatedelay(5);
int i = obj.getplayer();
int j = obj.getteleporter();
if (INBOUNDS_VEC(i, obj.entities))
{
if (INBOUNDS_VEC(j, obj.entities))
{
obj.entities[i].xp = obj.entities[j].xp+44;
obj.entities[i].yp = obj.entities[j].yp+44;
obj.entities[i].lerpoldxp = obj.entities[i].xp;
obj.entities[i].lerpoldyp = obj.entities[i].yp;
obj.entities[j].tile = 2;
obj.entities[j].colour = 101;
}
obj.entities[i].invis = false;
obj.entities[i].dir = 1;
obj.entities[i].colour = graphics.crewcolour(lastsaved);
obj.entities[i].ay = -6;
obj.entities[i].ax = 6;
obj.entities[i].vy = -6;
obj.entities[i].vx = 6;
}
break;
}
case 4073:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 10;
}
break;
}
case 4074:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 10;
}
break;
}
case 4075:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 8;
}
break;
}
case 4076:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 6;
}
break;
}
case 4077:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 3;
}
break;
}
case 4078:
{
incstate();
setstatedelay(15);
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 1;
}
break;
}
case 4079:
setstate(0);
startscript = true;
newscript = "finallevel_teleporter";
break;
case 4080:
//Activating a teleporter (default appear)
incstate();
setstatedelay(15);
flashlight = 5;
screenshake = 90;
music.playef(Sound_FLASH);
break;
case 4081:
//Activating a teleporter 2
incstate();
setstatedelay(0);
flashlight = 5;
screenshake = 0;
music.playef(Sound_TELEPORT);
break;
case 4082:
{
//Activating a teleporter 2
incstate();
setstatedelay(5);
int i = obj.getplayer();
int j = obj.getteleporter();
if (INBOUNDS_VEC(i, obj.entities))
{
if (INBOUNDS_VEC(j, obj.entities))
{
obj.entities[i].xp = obj.entities[j].xp+44;
obj.entities[i].yp = obj.entities[j].yp+44;
obj.entities[i].lerpoldxp = obj.entities[i].xp;
obj.entities[i].lerpoldyp = obj.entities[i].yp;
obj.entities[j].tile = 2;
obj.entities[j].colour = 101;
}
obj.entities[i].colour = 0;
obj.entities[i].invis = false;
obj.entities[i].dir = 1;
obj.entities[i].ay = -6;
obj.entities[i].ax = 6;
obj.entities[i].vy = -6;
obj.entities[i].vx = 6;
}
break;
}
case 4083:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 10;
}
break;
}
case 4084:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 10;
}
break;
}
case 4085:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 8;
}
break;
}
case 4086:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 6;
}
break;
}
case 4087:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 3;
}
break;
}
case 4088:
{
incstate();
setstatedelay(15);
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 1;
}
break;
}
case 4089:
startscript = true;
newscript = "gamecomplete_ending";
setstate(0);
break;
case 4090:
//Activating a teleporter (default appear)
incstate();
setstatedelay(15);
flashlight = 5;
screenshake = 90;
music.playef(Sound_FLASH);
break;
case 4091:
//Activating a teleporter 2
incstate();
setstatedelay(0);
flashlight = 5;
screenshake = 0;
music.playef(Sound_TELEPORT);
break;
case 4092:
{
//Activating a teleporter 2
incstate();
setstatedelay(5);
int i = obj.getplayer();
int j = obj.getteleporter();
if (INBOUNDS_VEC(i, obj.entities))
{
if (INBOUNDS_VEC(j, obj.entities))
{
obj.entities[i].xp = obj.entities[j].xp+44;
obj.entities[i].yp = obj.entities[j].yp+44;
obj.entities[i].lerpoldxp = obj.entities[i].xp;
obj.entities[i].lerpoldyp = obj.entities[i].yp;
obj.entities[j].tile = 2;
obj.entities[j].colour = 101;
}
obj.entities[i].colour = 0;
obj.entities[i].invis = false;
obj.entities[i].dir = 1;
obj.entities[i].ay = -6;
obj.entities[i].ax = 6;
obj.entities[i].vy = -6;
obj.entities[i].vx = 6;
}
break;
}
case 4093:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 10;
}
break;
}
case 4094:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 10;
}
break;
}
case 4095:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 8;
}
break;
}
case 4096:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 6;
}
break;
}
case 4097:
{
incstate();
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 3;
}
break;
}
case 4098:
{
incstate();
setstatedelay(15);
int i = obj.getplayer();
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].xp += 1;
}
break;
}
case 4099:
if (nocutscenes)
{
startscript = true;
newscript = "levelonecompleteskip";
}
else
{
startscript = true;
newscript = "levelonecomplete_ending";
}
setstate(0);
break;
}
}
}
void Game::gethardestroom(void)
{
if (currentroomdeaths > hardestroomdeaths)
{
hardestroomdeaths = currentroomdeaths;
hardestroom_x = roomx;
hardestroom_y = roomy;
hardestroom_finalstretch = map.finalstretch;
if (map.roomname[0] == '\0')
{
hardestroom = map.hiddenname;
hardestroom_specialname = true;
}
else
{
hardestroom = map.roomname;
hardestroom_specialname = map.roomname_special;
}
}
}
void Game::deletestats(void)
{
if (!FILESYSTEM_delete("saves/unlock.vvv"))
{
vlog_error("Error deleting saves/unlock.vvv");
}
else
{
for (int i = 0; i < numunlock; i++)
{
unlock[i] = false;
unlocknotify[i] = false;
}
for (int i = 0; i < numtrials; i++)
{
besttimes[i] = -1;
bestframes[i] = -1;
besttrinkets[i] = -1;
bestlives[i] = -1;
bestrank[i] = -1;
}
swnrecord = 0;
swnbestrank = 0;
bestgamedeaths = -1;
#ifndef MAKEANDPLAY
graphics.setflipmode = false;
#endif
stat_trinkets = 0;
}
}
void Game::deletesettings(void)
{
if (!FILESYSTEM_delete("saves/settings.vvv"))
{
vlog_error("Error deleting saves/settings.vvv");
}
}
void Game::unlocknum( int t )
{
#ifdef MAKEANDPLAY
UNUSED(t);
#else
if (map.custommode)
{
//Don't let custom levels unlock things!
return;
}
unlock[t] = true;
savestatsandsettings();
#endif
}
static bool stats_loaded = false;
void Game::loadstats(struct ScreenSettings* screen_settings)
{
tinyxml2::XMLDocument doc;
tinyxml2::XMLHandle hDoc(&doc);
tinyxml2::XMLElement* pElem;
tinyxml2::XMLElement* dataNode;
stats_loaded = true;
if (!FILESYSTEM_loadTiXml2Document("saves/unlock.vvv", doc))
{
// Save unlock.vvv only. Maybe we have a settings.vvv laying around too,
// and we don't want to overwrite that!
savestats(screen_settings);
return;
}
if (doc.Error())
{
vlog_error("Error parsing unlock.vvv: %s", doc.ErrorStr());
return;
}
dataNode = hDoc
.FirstChildElement()
.FirstChildElement("Data")
.FirstChildElement()
.ToElement();
for (pElem = dataNode; pElem != NULL; pElem=pElem->NextSiblingElement())
{
const char* pKey = pElem->Value();
const char* pText = pElem->GetText() ;
if (pText == NULL)
{
pText = "";
}
LOAD_ARRAY(unlock)
LOAD_ARRAY(unlocknotify)
LOAD_ARRAY(besttimes)
LOAD_ARRAY(bestframes)
LOAD_ARRAY(besttrinkets)
LOAD_ARRAY(bestlives)
LOAD_ARRAY(bestrank)
if (SDL_strcmp(pKey, "bestgamedeaths") == 0)
{
bestgamedeaths = help.Int(pText);
}
if (SDL_strcmp(pKey, "stat_trinkets") == 0)
{
stat_trinkets = help.Int(pText);
}
if (SDL_strcmp(pKey, "swnbestrank") == 0)
{
swnbestrank = help.Int(pText);
}
if (SDL_strcmp(pKey, "swnrecord") == 0)
{
swnrecord = help.Int(pText);
}
}
deserializesettings(dataNode, screen_settings);
}
void Game::deserializesettings(tinyxml2::XMLElement* dataNode, struct ScreenSettings* screen_settings)
{
// Don't duplicate controller buttons!
controllerButton_flip.clear();
controllerButton_map.clear();
controllerButton_esc.clear();
controllerButton_restart.clear();
controllerButton_interact.clear();
for (tinyxml2::XMLElement* pElem = dataNode;
pElem != NULL;
pElem = pElem->NextSiblingElement())
{
const char* pKey = pElem->Value();
const char* pText = pElem->GetText();
if (pText == NULL)
{
pText = "";
}
if (SDL_strcmp(pKey, "fullscreen") == 0)
{
screen_settings->fullscreen = help.Int(pText);
}
if (SDL_strcmp(pKey, "stretch") == 0)
{
int mode = help.Int(pText);
if (mode < 0 || mode >= NUM_SCALING_MODES)
{
/* Pick a sane default. */
mode = SCALING_INTEGER;
}
screen_settings->scalingMode = mode;
}
if (SDL_strcmp(pKey, "useLinearFilter") == 0)
{
screen_settings->linearFilter = help.Int(pText);
}
if (SDL_strcmp(pKey, "window_display") == 0)
{
screen_settings->windowDisplay = help.Int(pText);
}
if (SDL_strcmp(pKey, "window_width") == 0)
{
screen_settings->windowWidth = help.Int(pText);
}
if (SDL_strcmp(pKey, "window_height") == 0)
{
screen_settings->windowHeight = help.Int(pText);
}
if (SDL_strcmp(pKey, "noflashingmode") == 0)
{
noflashingmode = help.Int(pText);
}
if (SDL_strcmp(pKey, "colourblindmode") == 0)
{
colourblindmode = help.Int(pText);
}
if (SDL_strcmp(pKey, "setflipmode") == 0)
{
graphics.setflipmode = help.Int(pText);
}
if (SDL_strcmp(pKey, "invincibility") == 0)
{
map.invincibility = help.Int(pText);
}
if (SDL_strcmp(pKey, "slowdown") == 0)
{
slowdown = help.Int(pText);
}
if (SDL_strcmp(pKey, "advanced_smoothing") == 0)
{
screen_settings->badSignal = help.Int(pText);
}
if (SDL_strcmp(pKey, "usingmmmmmm") == 0)
{
music.usingmmmmmm = (bool) help.Int(pText);
}
if (SDL_strcmp(pKey, "ghostsenabled") == 0)
{
ghostsenabled = help.Int(pText);
}
if (SDL_strcmp(pKey, "skipfakeload") == 0)
{
skipfakeload = help.Int(pText);
}
if (SDL_strcmp(pKey, "disablepause") == 0)
{
disablepause = help.Int(pText);
}
if (SDL_strcmp(pKey, "disableaudiopause") == 0)
{
disableaudiopause = help.Int(pText);
}
if (SDL_strcmp(pKey, "over30mode") == 0)
{
over30mode = help.Int(pText);
}
if (SDL_strcmp(pKey, "inputdelay") == 0)
{
inputdelay = help.Int(pText);
}
if (SDL_strcmp(pKey, "glitchrunnermode") == 0)
{
GlitchrunnerMode_set(GlitchrunnerMode_string_to_enum(pText));
}
if (SDL_strcmp(pKey, "showingametimer") == 0)
{
showingametimer = help.Int(pText);
}
if (SDL_strcmp(pKey, "vsync") == 0)
{
screen_settings->useVsync = help.Int(pText);
}
if (SDL_strcmp(pKey, "notextoutline") == 0)
{
graphics.notextoutline = help.Int(pText);
}
if (SDL_strcmp(pKey, "translucentroomname") == 0)
{
graphics.translucentroomname = help.Int(pText);
}
if (SDL_strcmp(pKey, "musicvolume") == 0)
{
music.user_music_volume = help.Int(pText);
}
if (SDL_strcmp(pKey, "soundvolume") == 0)
{
music.user_sound_volume = help.Int(pText);
}
if (SDL_strcmp(pKey, "separate_interact") == 0)
{
separate_interact = help.Int(pText);
}
if (SDL_strcmp(pKey, "flipButton") == 0)
{
SDL_GameControllerButton newButton;
if (GetButtonFromString(pText, &newButton))
{
controllerButton_flip.push_back(newButton);
}
}
if (SDL_strcmp(pKey, "enterButton") == 0)
{
SDL_GameControllerButton newButton;
if (GetButtonFromString(pText, &newButton))
{
controllerButton_map.push_back(newButton);
}
}
if (SDL_strcmp(pKey, "escButton") == 0)
{
SDL_GameControllerButton newButton;
if (GetButtonFromString(pText, &newButton))
{
controllerButton_esc.push_back(newButton);
}
}
if (SDL_strcmp(pKey, "restartButton") == 0)
{
SDL_GameControllerButton newButton;
if (GetButtonFromString(pText, &newButton))
{
controllerButton_restart.push_back(newButton);
}
}
if (SDL_strcmp(pKey, "interactButton") == 0)
{
SDL_GameControllerButton newButton;
if (GetButtonFromString(pText, &newButton))
{
controllerButton_interact.push_back(newButton);
}
}
if (SDL_strcmp(pKey, "controllerSensitivity") == 0)
{
key.sensitivity = help.Int(pText);
}
if (SDL_strcmp(pKey, "lang") == 0)
{
loc::lang = std::string(pText);
}
if (SDL_strcmp(pKey, "lang_set") == 0)
{
loc::lang_set = help.Int(pText);
}
if (SDL_strcmp(pKey, "english_sprites") == 0)
{
loc::english_sprites = help.Int(pText);
}
if (SDL_strcmp(pKey, "new_level_font") == 0)
{
loc::new_level_font = std::string(pText);
}
if (SDL_strcmp(pKey, "roomname_translator") == 0 && loc::show_translator_menu)
{
roomname_translator::set_enabled(help.Int(pText));
}
}
setdefaultcontrollerbuttons();
}
bool Game::savestats(bool sync /*= true*/)
{
struct ScreenSettings screen_settings;
SDL_zero(screen_settings);
gameScreen.GetSettings(&screen_settings);
return savestats(&screen_settings, sync);
}
bool Game::savestats(const struct ScreenSettings* screen_settings, bool sync /*= true*/)
{
tinyxml2::XMLDocument doc;
bool already_exists;
if (!stats_loaded)
{
vlog_warn("Stats not loaded! Not writing unlock.vvv.");
return false;
}
already_exists = FILESYSTEM_loadTiXml2Document("saves/unlock.vvv", doc);
if (!already_exists)
{
vlog_info("No unlock.vvv found. Creating new file");
}
else if (doc.Error())
{
vlog_error("Error parsing existing unlock.vvv: %s", doc.ErrorStr());
vlog_info("Creating new unlock.vvv");
}
xml::update_declaration(doc);
tinyxml2::XMLElement * root = xml::update_element(doc, "Save");
xml::update_comment(root, " Save file " );
tinyxml2::XMLElement * dataNode = xml::update_element(root, "Data");
std::string s_unlock;
for(size_t i = 0; i < SDL_arraysize(unlock); i++ )
{
s_unlock += help.String(unlock[i]) + ",";
}
xml::update_tag(dataNode, "unlock", s_unlock.c_str());
std::string s_unlocknotify;
for(size_t i = 0; i < SDL_arraysize(unlocknotify); i++ )
{
s_unlocknotify += help.String(unlocknotify[i]) + ",";
}
xml::update_tag(dataNode, "unlocknotify", s_unlocknotify.c_str());
std::string s_besttimes;
for(size_t i = 0; i < SDL_arraysize(besttimes); i++ )
{
s_besttimes += help.String(besttimes[i]) + ",";
}
xml::update_tag(dataNode, "besttimes", s_besttimes.c_str());
std::string s_bestframes;
for (size_t i = 0; i < SDL_arraysize(bestframes); i++)
{
s_bestframes += help.String(bestframes[i]) + ",";
}
xml::update_tag(dataNode, "bestframes", s_bestframes.c_str());
std::string s_besttrinkets;
for(size_t i = 0; i < SDL_arraysize(besttrinkets); i++ )
{
s_besttrinkets += help.String(besttrinkets[i]) + ",";
}
xml::update_tag(dataNode, "besttrinkets", s_besttrinkets.c_str());
std::string s_bestlives;
for(size_t i = 0; i < SDL_arraysize(bestlives); i++ )
{
s_bestlives += help.String(bestlives[i]) + ",";
}
xml::update_tag(dataNode, "bestlives", s_bestlives.c_str());
std::string s_bestrank;
for(size_t i = 0; i < SDL_arraysize(bestrank); i++ )
{
s_bestrank += help.String(bestrank[i]) + ",";
}
xml::update_tag(dataNode, "bestrank", s_bestrank.c_str());
xml::update_tag(dataNode, "bestgamedeaths", bestgamedeaths);
xml::update_tag(dataNode, "stat_trinkets", stat_trinkets);
xml::update_tag(dataNode, "swnbestrank", swnbestrank);
xml::update_tag(dataNode, "swnrecord", swnrecord);
serializesettings(dataNode, screen_settings);
return FILESYSTEM_saveTiXml2Document("saves/unlock.vvv", doc, sync);
}
bool Game::savestatsandsettings(void)
{
const bool stats_saved = savestats(false);
const bool settings_saved = savesettings();
return stats_saved && settings_saved; // Not the same as `savestats() && savesettings()`!
}
void Game::savestatsandsettings_menu(void)
{
// Call Game::savestatsandsettings(), but upon failure, go to the save error screen
if (!savestatsandsettings() && !silence_settings_error)
{
createmenu(Menu::errorsavingsettings);
map.nexttowercolour();
}
}
void Game::serializesettings(tinyxml2::XMLElement* dataNode, const struct ScreenSettings* screen_settings)
{
tinyxml2::XMLDocument& doc = xml::get_document(dataNode);
xml::update_tag(dataNode, "fullscreen", (int) screen_settings->fullscreen);
xml::update_tag(dataNode, "stretch", screen_settings->scalingMode);
xml::update_tag(dataNode, "useLinearFilter", (int) screen_settings->linearFilter);
xml::update_tag(dataNode, "window_display", screen_settings->windowDisplay);
xml::update_tag(dataNode, "window_width", screen_settings->windowWidth);
xml::update_tag(dataNode, "window_height", screen_settings->windowHeight);
xml::update_tag(dataNode, "noflashingmode", noflashingmode);
xml::update_tag(dataNode, "colourblindmode", colourblindmode);
xml::update_tag(dataNode, "setflipmode", graphics.setflipmode);
xml::update_tag(dataNode, "invincibility", map.invincibility);
xml::update_tag(dataNode, "slowdown", slowdown);
xml::update_tag(dataNode, "advanced_smoothing", (int) screen_settings->badSignal);
xml::update_tag(dataNode, "usingmmmmmm", music.usingmmmmmm);
xml::update_tag(dataNode, "ghostsenabled", (int) ghostsenabled);
xml::update_tag(dataNode, "skipfakeload", (int) skipfakeload);
xml::update_tag(dataNode, "disablepause", (int) disablepause);
xml::update_tag(dataNode, "disableaudiopause", (int) disableaudiopause);
xml::update_tag(dataNode, "notextoutline", (int) graphics.notextoutline);
xml::update_tag(dataNode, "translucentroomname", (int) graphics.translucentroomname);
xml::update_tag(dataNode, "over30mode", (int) over30mode);
xml::update_tag(dataNode, "inputdelay", (int) inputdelay);
xml::update_tag(
dataNode,
"glitchrunnermode",
GlitchrunnerMode_enum_to_string(GlitchrunnerMode_get())
);
xml::update_tag(dataNode, "showingametimer", (int) showingametimer);
xml::update_tag(dataNode, "vsync", (int) screen_settings->useVsync);
xml::update_tag(dataNode, "musicvolume", music.user_music_volume);
xml::update_tag(dataNode, "soundvolume", music.user_sound_volume);
xml::update_tag(dataNode, "separate_interact", (int) separate_interact);
// Delete all controller buttons we had previously.
// dataNode->FirstChildElement() shouldn't be NULL at this point...
// we've already added a bunch of elements
for (tinyxml2::XMLElement* element = dataNode->FirstChildElement();
element != NULL;
/* Increment code handled separately */)
{
const char* name = element->Name();
if (SDL_strcmp(name, "flipButton") == 0
|| SDL_strcmp(name, "enterButton") == 0
|| SDL_strcmp(name, "escButton") == 0
|| SDL_strcmp(name, "restartButton") == 0
|| SDL_strcmp(name, "interactButton") == 0)
{
// Can't just doc.DeleteNode(element) and then go to next,
// element->NextSiblingElement() will be NULL.
// Instead, store pointer of element we want to delete. Then
// increment `element`. And THEN delete the element.
tinyxml2::XMLElement* delete_this = element;
element = element->NextSiblingElement();
doc.DeleteNode(delete_this);
continue;
}
element = element->NextSiblingElement();
}
// Now add them
for (size_t i = 0; i < controllerButton_flip.size(); i += 1)
{
tinyxml2::XMLElement* msg = doc.NewElement("flipButton");
msg->LinkEndChild(doc.NewText(help.String((int) controllerButton_flip[i]).c_str()));
dataNode->LinkEndChild(msg);
}
for (size_t i = 0; i < controllerButton_map.size(); i += 1)
{
tinyxml2::XMLElement* msg = doc.NewElement("enterButton");
msg->LinkEndChild(doc.NewText(help.String((int) controllerButton_map[i]).c_str()));
dataNode->LinkEndChild(msg);
}
for (size_t i = 0; i < controllerButton_esc.size(); i += 1)
{
tinyxml2::XMLElement* msg = doc.NewElement("escButton");
msg->LinkEndChild(doc.NewText(help.String((int) controllerButton_esc[i]).c_str()));
dataNode->LinkEndChild(msg);
}
for (size_t i = 0; i < controllerButton_restart.size(); i += 1)
{
tinyxml2::XMLElement* msg = doc.NewElement("restartButton");
msg->LinkEndChild(doc.NewText(help.String((int) controllerButton_restart[i]).c_str()));
dataNode->LinkEndChild(msg);
}
for (size_t i = 0; i < controllerButton_interact.size(); i += 1)
{
tinyxml2::XMLElement* msg = doc.NewElement("interactButton");
msg->LinkEndChild(doc.NewText(help.String((int) controllerButton_interact[i]).c_str()));
dataNode->LinkEndChild(msg);
}
xml::update_tag(dataNode, "controllerSensitivity", key.sensitivity);
xml::update_tag(dataNode, "lang", loc::lang.c_str());
xml::update_tag(dataNode, "lang_set", (int) loc::lang_set);
xml::update_tag(dataNode, "english_sprites", (int) loc::english_sprites);
xml::update_tag(dataNode, "new_level_font", loc::new_level_font.c_str());
xml::update_tag(dataNode, "roomname_translator", (int) roomname_translator::enabled);
}
static bool settings_loaded = false;
void Game::loadsettings(struct ScreenSettings* screen_settings)
{
tinyxml2::XMLDocument doc;
tinyxml2::XMLHandle hDoc(&doc);
tinyxml2::XMLElement* dataNode;
settings_loaded = true;
if (!FILESYSTEM_loadTiXml2Document("saves/settings.vvv", doc))
{
savesettings(screen_settings);
return;
}
if (doc.Error())
{
vlog_error("Error parsing settings.vvv: %s", doc.ErrorStr());
return;
}
dataNode = hDoc
.FirstChildElement()
.FirstChildElement("Data")
.FirstChildElement()
.ToElement();
deserializesettings(dataNode, screen_settings);
}
bool Game::savesettings(void)
{
struct ScreenSettings screen_settings;
SDL_zero(screen_settings);
gameScreen.GetSettings(&screen_settings);
return savesettings(&screen_settings);
}
bool Game::savesettings(const struct ScreenSettings* screen_settings)
{
tinyxml2::XMLDocument doc;
bool already_exists;
if (!settings_loaded)
{
vlog_warn("Settings not loaded! Not writing settings.vvv.");
return false;
}
already_exists = FILESYSTEM_loadTiXml2Document("saves/settings.vvv", doc);
if (!already_exists)
{
vlog_info("No settings.vvv found. Creating new file");
}
else if (doc.Error())
{
vlog_error("Error parsing existing settings.vvv: %s", doc.ErrorStr());
vlog_info("Creating new settings.vvv");
}
xml::update_declaration(doc);
tinyxml2::XMLElement* root = xml::update_element(doc, "Settings");
xml::update_comment(root, " Settings (duplicated from unlock.vvv) ");
tinyxml2::XMLElement* dataNode = xml::update_element(root, "Data");
serializesettings(dataNode, screen_settings);
return FILESYSTEM_saveTiXml2Document("saves/settings.vvv", doc);
}
void Game::customstart(void)
{
jumpheld = true;
savex = edsavex;
savey = edsavey;
saverx = edsaverx;
savery = edsavery;
savegc = edsavegc;
savedir = edsavedir; //Worldmap Start
savepoint = 0;
gravitycontrol = savegc;
setstate(0);
deathseq = -1;
lifeseq = 0;
}
void Game::start(void)
{
jumpheld = true;
savex = 232;
savey = 113;
saverx = 104;
savery = 110;
savegc = 0;
savedir = 1; //Worldmap Start
savepoint = 0;
gravitycontrol = savegc;
setstate(0);
deathseq = -1;
lifeseq = 0;
if (!nocutscenes)
{
music.play(Music_PAUSE);
}
}
void Game::deathsequence(void)
{
int i;
if (supercrewmate && scmhurt)
{
i = obj.getscm();
}
else
{
i = obj.getplayer();
}
if (INBOUNDS_VEC(i, obj.entities))
{
obj.entities[i].colour = 1;
obj.entities[i].invis = false;
}
if (deathseq == 30)
{
if (nodeathmode)
{
music.fadeout();
gameoverdelay = 60;
/* Fix a bug being able to play music on the Game Over screen */
music.nicefade = false;
}
deathcounts++;
music.playef(Sound_CRY);
if (INBOUNDS_VEC(i, obj.entities) && !noflashingmode)
{
obj.entities[i].invis = true;
}
if (map.finalmode)
{
if (roomx - 41 >= 0 && roomx - 41 < 20 && roomy - 48 >= 0 && roomy - 48 < 20)
{
map.roomdeathsfinal[roomx - 41 + (20 * (roomy - 48))]++;
currentroomdeaths = map.roomdeathsfinal[roomx - 41 + (20 * (roomy - 48))];
}
}
else
{
if (roomx - 100 >= 0 && roomx - 100 < 20 && roomy - 100 >= 0 && roomy - 100 < 20)
{
map.roomdeaths[roomx - 100 + (20*(roomy - 100))]++;
currentroomdeaths = map.roomdeaths[roomx - 100 + (20 * (roomy - 100))];
}
}
}
if (INBOUNDS_VEC(i, obj.entities) && !noflashingmode)
{
if (deathseq == 25) obj.entities[i].invis = true;
if (deathseq == 20) obj.entities[i].invis = true;
if (deathseq == 16) obj.entities[i].invis = true;
if (deathseq == 14) obj.entities[i].invis = true;
if (deathseq == 12) obj.entities[i].invis = true;
if (deathseq < 10) obj.entities[i].invis = true;
}
if (!nodeathmode)
{
if (INBOUNDS_VEC(i, obj.entities) && deathseq <= 1) obj.entities[i].invis = false;
}
else
{
gameoverdelay--;
}
}
void Game::startspecial( int t )
{
jumpheld = true;
switch(t)
{
case 0: //Secret Lab
savex = 104;
savey = 169;
saverx = 118;
savery = 106;
savegc = 0;
savedir = 1;
break;
case 1: //Intermission 1 (any)
savex = 80;
savey = 57;
saverx = 41;
savery = 56;
savegc = 0;
savedir = 0;
break;
default:
savex = 232;
savey = 113;
saverx = 104;
savery = 110;
savegc = 0;
savedir = 1; //Worldmap Start
break;
}
savepoint = 0;
gravitycontrol = savegc;
setstate(0);
deathseq = -1;
lifeseq = 0;
}
void Game::starttrial( int t )
{
jumpheld = true;
switch(t)
{
case 0: //Space Station 1
savex = 200;
savey = 161;
saverx = 113;
savery = 105;
savegc = 0;
savedir = 1;
break;
case 1: //Lab
savex = 191;
savey = 33;
saverx = 102;
savery = 116;
savegc = 0;
savedir = 1;
break;
case 2: //Tower
savex = 84;
savey = 193, saverx = 108;
savery = 109;
savegc = 0;
savedir = 1;
break;
case 3: //Space Station 2
savex = 148;
savey = 38;
saverx = 112;
savery = 114;
savegc = 1;
savedir = 0;
break;
case 4: //Warp
savex = 52;
savey = 73;
saverx = 114;
savery = 101;
savegc = 0;
savedir = 1;
break;
case 5: //Final
savex = 101;
savey = 113;
saverx = 46;
savery = 54;
savegc = 0;
savedir = 1;
break;
default:
savex = 232;
savey = 113;
saverx = 104;
savery = 110;
savegc = 0;
savedir = 1; //Worldmap Start
break;
}
savepoint = 0;
gravitycontrol = savegc;
setstate(0);
deathseq = -1;
lifeseq = 0;
}
void Game::loadquick(void)
{
tinyxml2::XMLDocument doc;
if (!FILESYSTEM_loadTiXml2Document("saves/qsave.vvv", doc)) return;
readmaingamesave("qsave.vvv", doc);
}
void Game::readmaingamesave(const char* savename, tinyxml2::XMLDocument& doc)
{
tinyxml2::XMLHandle hDoc(&doc);
tinyxml2::XMLElement* pElem;
if (doc.Error())
{
vlog_error("Error parsing %s: %s", savename, doc.ErrorStr());
return;
}
/* Even if we want the default hardest room to be Welcome Aboard, there are pre-2.4
* saves with JUST <hardestroom> which should take priority over the coords */
hardestroom_x = -1;
hardestroom_y = -1;
hardestroom_specialname = false;
hardestroom_finalstretch = false;
for (pElem = hDoc
.FirstChildElement()
.FirstChildElement("Data")
.FirstChildElement()
.ToElement();
pElem != NULL;
pElem = pElem->NextSiblingElement())
{
const char* pKey = pElem->Value();
const char* pText = pElem->GetText();
if(pText == NULL)
{
pText = "";
}
LOAD_ARRAY_RENAME(worldmap, map.explored)
LOAD_ARRAY_RENAME(flags, obj.flags)
LOAD_ARRAY(crewstats)
LOAD_ARRAY_RENAME(collect, obj.collect)
if (SDL_strcmp(pKey, "finalmode") == 0)
{
map.finalmode = help.Int(pText);
}
if (SDL_strcmp(pKey, "finalstretch") == 0)
{
map.finalstretch = help.Int(pText);
}
if (SDL_strcmp(pKey, "savex") == 0)
{
savex = help.Int(pText);
}
else if (SDL_strcmp(pKey, "savey") == 0)
{
savey = help.Int(pText);
}
else if (SDL_strcmp(pKey, "saverx") == 0)
{
saverx = help.Int(pText);
}
else if (SDL_strcmp(pKey, "savery") == 0)
{
savery = help.Int(pText);
}
else if (SDL_strcmp(pKey, "savegc") == 0)
{
savegc = help.Int(pText);
}
else if (SDL_strcmp(pKey, "savedir") == 0)
{
savedir= help.Int(pText);
}
else if (SDL_strcmp(pKey, "savepoint") == 0)
{
savepoint = help.Int(pText);
}
else if (SDL_strcmp(pKey, "companion") == 0)
{
companion = help.Int(pText);
}
else if (SDL_strcmp(pKey, "lastsaved") == 0)
{
lastsaved = help.Int(pText);
}
else if (SDL_strcmp(pKey, "teleportscript") == 0)
{
teleportscript = pText;
}
else if (SDL_strcmp(pKey, "supercrewmate") == 0)
{
supercrewmate = help.Int(pText);
}
else if (SDL_strcmp(pKey, "scmprogress") == 0)
{
scmprogress = help.Int(pText);
}
else if (SDL_strcmp(pKey, "frames") == 0)
{
frames = help.Int(pText);
}
else if (SDL_strcmp(pKey, "seconds") == 0)
{
seconds = help.Int(pText);
}
else if (SDL_strcmp(pKey, "minutes") == 0)
{
minutes = help.Int(pText);
}
else if (SDL_strcmp(pKey, "hours") == 0)
{
hours = help.Int(pText);
}
else if (SDL_strcmp(pKey, "deathcounts") == 0)
{
deathcounts = help.Int(pText);
}
else if (SDL_strcmp(pKey, "totalflips") == 0)
{
totalflips = help.Int(pText);
}
else if (SDL_strcmp(pKey, "hardestroom") == 0)
{
hardestroom = pText;
}
else if (SDL_strcmp(pKey, "hardestroomdeaths") == 0)
{
hardestroomdeaths = help.Int(pText);
}
else if (SDL_strcmp(pKey, "hardestroom_x") == 0)
{
hardestroom_x = help.Int(pText);
}
else if (SDL_strcmp(pKey, "hardestroom_y") == 0)
{
hardestroom_y = help.Int(pText);
}
else if (SDL_strcmp(pKey, "hardestroom_specialname") == 0)
{
hardestroom_specialname = help.Int(pText);
}
else if (SDL_strcmp(pKey, "hardestroom_finalstretch") == 0)
{
hardestroom_finalstretch = help.Int(pText);
}
else if (SDL_strcmp(pKey, "currentsong") == 0)
{
int song = help.Int(pText);
if (song != -1)
{
music.play(song);
}
}
else if (SDL_strcmp(pKey, "showtargets") == 0)
{
map.showtargets = help.Int(pText);
}
}
if (map.finalmode)
{
map.final_colormode = false;
map.final_mapcol = 0;
map.final_colorframe = 0;
}
if (map.finalstretch)
{
map.finalstretch = true;
map.final_colormode = true;
map.final_mapcol = 0;
map.final_colorframe = 1;
}
map.showteleporters = true;
if(obj.flags[12]) map.showtargets = true;
if (obj.flags[42]) map.showtrinkets = true;
}
void Game::customloadquick(const std::string& savfile)
{
tinyxml2::XMLDocument doc;
tinyxml2::XMLHandle hDoc(&doc);
tinyxml2::XMLElement* pElem;
std::string levelfile;
if (cliplaytest)
{
savex = playx;
savey = playy;
saverx = playrx;
savery = playry;
savegc = playgc;
if (playmusic > -1)
{
music.play(playmusic);
}
return;
}
levelfile = savfile.substr(7);
if (!FILESYSTEM_loadTiXml2Document(("saves/"+levelfile+".vvv").c_str(), doc))
{
vlog_error("%s.vvv not found", levelfile.c_str());
return;
}
if (doc.Error())
{
vlog_error("Error parsing %s.vvv: %s", levelfile.c_str(), doc.ErrorStr());
return;
}
// Like readmaingamesave(...), old saves have just <hardestroom>
hardestroom_x = -1;
hardestroom_y = -1;
hardestroom_specialname = false;
hardestroom_finalstretch = false;
for (pElem = hDoc
.FirstChildElement()
.FirstChildElement("Data")
.FirstChildElement()
.ToElement();
pElem != NULL;
pElem = pElem->NextSiblingElement())
{
const char* pKey = pElem->Value();
const char* pText = pElem->GetText() ;
if(pText == NULL)
{
pText = "";
}
LOAD_ARRAY_RENAME(worldmap, map.explored)
LOAD_ARRAY_RENAME(flags, obj.flags)
LOAD_ARRAY_RENAME(moods, obj.customcrewmoods)
LOAD_ARRAY(crewstats)
LOAD_ARRAY_RENAME(collect, obj.collect)
LOAD_ARRAY_RENAME(customcollect, obj.customcollect)
if (SDL_strcmp(pKey, "finalmode") == 0)
{
map.finalmode = help.Int(pText);
}
if (SDL_strcmp(pKey, "finalstretch") == 0)
{
map.finalstretch = help.Int(pText);
}
if (map.finalmode)
{
map.final_colormode = false;
map.final_mapcol = 0;
map.final_colorframe = 0;
}
if (map.finalstretch)
{
map.finalstretch = true;
map.final_colormode = true;
map.final_mapcol = 0;
map.final_colorframe = 1;
}
if (SDL_strcmp(pKey, "savex") == 0)
{
savex = help.Int(pText);
}
else if (SDL_strcmp(pKey, "savey") == 0)
{
savey = help.Int(pText);
}
else if (SDL_strcmp(pKey, "saverx") == 0)
{
saverx = help.Int(pText);
}
else if (SDL_strcmp(pKey, "savery") == 0)
{
savery = help.Int(pText);
}
else if (SDL_strcmp(pKey, "savegc") == 0)
{
savegc = help.Int(pText);
}
else if (SDL_strcmp(pKey, "savedir") == 0)
{
savedir= help.Int(pText);
}
else if (SDL_strcmp(pKey, "savepoint") == 0)
{
savepoint = help.Int(pText);
}
else if (SDL_strcmp(pKey, "savecolour") == 0)
{
savecolour = help.Int(pText);
}
else if (SDL_strcmp(pKey, "companion") == 0)
{
companion = help.Int(pText);
}
else if (SDL_strcmp(pKey, "lastsaved") == 0)
{
lastsaved = help.Int(pText);
}
else if (SDL_strcmp(pKey, "teleportscript") == 0)
{
teleportscript = pText;
}
else if (SDL_strcmp(pKey, "supercrewmate") == 0)
{
supercrewmate = help.Int(pText);
}
else if (SDL_strcmp(pKey, "scmprogress") == 0)
{
scmprogress = help.Int(pText);
}
else if (SDL_strcmp(pKey, "frames") == 0)
{
frames = help.Int(pText);
}
else if (SDL_strcmp(pKey, "seconds") == 0)
{
seconds = help.Int(pText);
}
else if (SDL_strcmp(pKey, "minutes") == 0)
{
minutes = help.Int(pText);
}
else if (SDL_strcmp(pKey, "hours") == 0)
{
hours = help.Int(pText);
}
else if (SDL_strcmp(pKey, "deathcounts") == 0)
{
deathcounts = help.Int(pText);
}
else if (SDL_strcmp(pKey, "totalflips") == 0)
{
totalflips = help.Int(pText);
}
else if (SDL_strcmp(pKey, "hardestroom") == 0)
{
hardestroom = pText;
}
else if (SDL_strcmp(pKey, "hardestroomdeaths") == 0)
{
hardestroomdeaths = help.Int(pText);
}
else if (SDL_strcmp(pKey, "hardestroom_x") == 0)
{
hardestroom_x = help.Int(pText);
}
else if (SDL_strcmp(pKey, "hardestroom_y") == 0)
{
hardestroom_y = help.Int(pText);
}
else if (SDL_strcmp(pKey, "hardestroom_specialname") == 0)
{
hardestroom_specialname = help.Int(pText);
}
else if (SDL_strcmp(pKey, "hardestroom_finalstretch") == 0)
{
hardestroom_finalstretch = help.Int(pText);
}
else if (SDL_strcmp(pKey, "currentsong") == 0)
{
int song = help.Int(pText);
if (song != -1)
{
music.play(song);
}
}
else if (SDL_strcmp(pKey, "lang_custom") == 0)
{
loc::lang_custom = pText;
if (pText[0] != '\0')
{
loc::loadtext_custom(NULL);
}
}
else if (SDL_strcmp(pKey, "showminimap") == 0)
{
map.customshowmm = help.Int(pText);
}
else if (SDL_strcmp(pKey, "disabletemporaryaudiopause") == 0)
{
disabletemporaryaudiopause = help.Int(pText);
}
else if (SDL_strcmp(pKey, "showtrinkets") == 0)
{
map.showtrinkets = help.Int(pText);
}
else if (SDL_strcmp(pKey, "roomname") == 0)
{
map.setroomname(pText);
map.roomnameset = true;
map.roomname_special = true;
}
}
}
static void loadthissummary(
const char* filename,
struct Game::Summary* summary,
tinyxml2::XMLDocument& doc
) {
tinyxml2::XMLHandle hDoc(&doc);
tinyxml2::XMLElement* pElem;
if (doc.Error())
{
vlog_error("Error parsing %s: %s", filename, doc.ErrorStr());
return;
}
summary->exists = true;
for (pElem = hDoc
.FirstChildElement()
.FirstChildElement("Data")
.FirstChildElement()
.ToElement();
pElem != NULL;
pElem = pElem->NextSiblingElement())
{
const char* pKey = pElem->Value();
const char* pText = pElem->GetText();
if (pText == NULL)
{
pText = "";
}
if (SDL_strcmp(pKey, "seconds") == 0)
{
summary->seconds = help.Int(pText);
}
else if (SDL_strcmp(pKey, "minutes") == 0)
{
summary->minutes = help.Int(pText);
}
else if (SDL_strcmp(pKey, "hours") == 0)
{
summary->hours = help.Int(pText);
}
else if (SDL_strcmp(pKey, "saverx") == 0)
{
summary->saverx = help.Int(pText);
}
else if (SDL_strcmp(pKey, "savery") == 0)
{
summary->savery = help.Int(pText);
}
else if (SDL_strcmp(pKey, "trinkets") == 0)
{
summary->trinkets = help.Int(pText);
}
LOAD_ARRAY_RENAME(crewstats, summary->crewstats)
}
}
void Game::loadsummary(void)
{
tinyxml2::XMLDocument doc;
SDL_zero(last_telesave);
SDL_zero(last_quicksave);
if (FILESYSTEM_loadTiXml2Document("saves/tsave.vvv", doc))
{
loadthissummary("tsave.vvv", &last_telesave, doc);
}
if (FILESYSTEM_loadTiXml2Document("saves/qsave.vvv", doc))
{
loadthissummary("qsave.vvv", &last_quicksave, doc);
}
}
void Game::initteleportermode(void)
{
//Set the teleporter variable to the right position!
teleport_to_teleporter = 0;
for (size_t i = 0; i < map.teleporters.size(); i++)
{
if (roomx == map.teleporters[i].x + 100 && roomy == map.teleporters[i].y + 100)
{
teleport_to_teleporter = i;
}
}
}
bool Game::savetele(void)
{
if (map.custommode || inspecial())
{
//Don't trash save data!
return false;
}
tinyxml2::XMLDocument doc;
bool already_exists = FILESYSTEM_loadTiXml2Document("saves/tsave.vvv", doc);
if (!already_exists)
{
vlog_info("No tsave.vvv found. Creating new file");
}
else if (doc.Error())
{
vlog_error("Error parsing existing tsave.vvv: %s", doc.ErrorStr());
vlog_info("Creating new tsave.vvv");
}
last_telesave = writemaingamesave(doc);
if(!FILESYSTEM_saveTiXml2Document("saves/tsave.vvv", doc))
{
vlog_error("Could Not Save game!");
vlog_error("Failed: %s%s", saveFilePath, "tsave.vvv");
return false;
}
vlog_info("Game saved");
return true;
}
bool Game::savequick(void)
{
if (map.custommode || inspecial())
{
//Don't trash save data!
return false;
}
tinyxml2::XMLDocument doc;
bool already_exists = FILESYSTEM_loadTiXml2Document("saves/qsave.vvv", doc);
if (!already_exists)
{
vlog_info("No qsave.vvv found. Creating new file");
}
else if (doc.Error())
{
vlog_error("Error parsing existing qsave.vvv: %s", doc.ErrorStr());
vlog_info("Creating new qsave.vvv");
}
last_quicksave = writemaingamesave(doc);
if(!FILESYSTEM_saveTiXml2Document("saves/qsave.vvv", doc))
{
vlog_error("Could Not Save game!");
vlog_error("Failed: %s%s", saveFilePath, "qsave.vvv");
return false;
}
vlog_info("Game saved");
return true;
}
// Returns summary of save
struct Game::Summary Game::writemaingamesave(tinyxml2::XMLDocument& doc)
{
//TODO make this code a bit cleaner.
struct Game::Summary summary;
SDL_zero(summary);
if (map.custommode || inspecial())
{
//Don't trash save data!
return summary;
}
xml::update_declaration(doc);
tinyxml2::XMLElement * root = xml::update_element(doc, "Save");
xml::update_comment(root, " Save file " );
tinyxml2::XMLElement * msgs = xml::update_element(root, "Data");
//Flags, map and stats
std::string mapExplored;
for(size_t i = 0; i < SDL_arraysize(map.explored); i++ )
{
mapExplored += help.String(map.explored[i]) + ",";
}
xml::update_tag(msgs, "worldmap", mapExplored.c_str());
std::string flags;
for(size_t i = 0; i < SDL_arraysize(obj.flags); i++ )
{
flags += help.String((int) obj.flags[i]) + ",";
}
xml::update_tag(msgs, "flags", flags.c_str());
std::string crewstatsString;
for(size_t i = 0; i < SDL_arraysize(crewstats); i++ )
{
crewstatsString += help.String(crewstats[i]) + ",";
}
xml::update_tag(msgs, "crewstats", crewstatsString.c_str());
std::string collect;
for(size_t i = 0; i < SDL_arraysize(obj.collect); i++ )
{
collect += help.String((int) obj.collect[i]) + ",";
}
xml::update_tag(msgs, "collect", collect.c_str());
//Position
xml::update_tag(msgs, "savex", savex);
xml::update_tag(msgs, "savey", savey);
xml::update_tag(msgs, "saverx", saverx);
xml::update_tag(msgs, "savery", savery);
xml::update_tag(msgs, "savegc", savegc);
xml::update_tag(msgs, "savedir", savedir);
xml::update_tag(msgs, "savepoint", savepoint);
int n_trinkets = trinkets();
xml::update_tag(msgs, "trinkets", n_trinkets);
//Special stats
if (music.nicefade)
{
xml::update_tag(msgs, "currentsong", music.nicechange);
}
else
{
xml::update_tag(msgs, "currentsong", music.currentsong);
}
xml::update_tag(msgs, "showtargets", (int) map.showtargets);
xml::update_tag(msgs, "teleportscript", teleportscript.c_str());
xml::update_tag(msgs, "companion", companion);
xml::update_tag(msgs, "lastsaved", lastsaved);
xml::update_tag(msgs, "supercrewmate", (int) supercrewmate);
xml::update_tag(msgs, "scmprogress", scmprogress);
xml::update_tag(msgs, "frames", frames);
xml::update_tag(msgs, "seconds", seconds);
xml::update_tag(msgs, "minutes", minutes);
xml::update_tag(msgs, "hours", hours);
xml::update_tag(msgs, "deathcounts", deathcounts);
xml::update_tag(msgs, "totalflips", totalflips);
xml::update_tag(msgs, "hardestroom", hardestroom.c_str());
xml::update_tag(msgs, "hardestroomdeaths", hardestroomdeaths);
xml::update_tag(msgs, "hardestroom_x", hardestroom_x);
xml::update_tag(msgs, "hardestroom_y", hardestroom_y);
xml::update_tag(msgs, "hardestroom_specialname", (int) hardestroom_specialname);
xml::update_tag(msgs, "hardestroom_finalstretch", (int) hardestroom_finalstretch);
xml::update_tag(msgs, "finalmode", (int) map.finalmode);
xml::update_tag(msgs, "finalstretch", (int) map.finalstretch);
std::string legacy_summary = std::string(map.currentarea(saverx, savery)) + ", " + timestring();
xml::update_tag(msgs, "summary", legacy_summary.c_str());
summary.exists = true;
summary.seconds = seconds;
summary.minutes = minutes;
summary.hours = hours;
summary.saverx = saverx;
summary.savery = savery;
summary.trinkets = n_trinkets;
SDL_memcpy(summary.crewstats, crewstats, sizeof(summary.crewstats));
return summary;
}
bool Game::customsavequick(const std::string& savfile)
{
const std::string levelfile = savfile.substr(7);
tinyxml2::XMLDocument doc;
bool already_exists = FILESYSTEM_loadTiXml2Document(("saves/" + levelfile + ".vvv").c_str(), doc);
if (!already_exists)
{
vlog_info("No %s.vvv found. Creating new file", levelfile.c_str());
}
else if (doc.Error())
{
vlog_error("Error parsing existing %s.vvv: %s", levelfile.c_str(), doc.ErrorStr());
vlog_info("Creating new %s.vvv", levelfile.c_str());
}
xml::update_declaration(doc);
tinyxml2::XMLElement * root = xml::update_element(doc, "Save");
xml::update_comment(root, " Save file ");
tinyxml2::XMLElement * msgs = xml::update_element(root, "Data");
//Flags, map and stats
std::string mapExplored;
for(size_t i = 0; i < SDL_arraysize(map.explored); i++ )
{
mapExplored += help.String(map.explored[i]) + ",";
}
xml::update_tag(msgs, "worldmap", mapExplored.c_str());
std::string flags;
for(size_t i = 0; i < SDL_arraysize(obj.flags); i++ )
{
flags += help.String((int) obj.flags[i]) + ",";
}
xml::update_tag(msgs, "flags", flags.c_str());
std::string moods;
for(size_t i = 0; i < SDL_arraysize(obj.customcrewmoods); i++ )
{
moods += help.String(obj.customcrewmoods[i]) + ",";
}
xml::update_tag(msgs, "moods", moods.c_str());
std::string crewstatsString;
for(size_t i = 0; i < SDL_arraysize(crewstats); i++ )
{
crewstatsString += help.String(crewstats[i]) + ",";
}
xml::update_tag(msgs, "crewstats", crewstatsString.c_str());
std::string collect;
for(size_t i = 0; i < SDL_arraysize(obj.collect); i++ )
{
collect += help.String((int) obj.collect[i]) + ",";
}
xml::update_tag(msgs, "collect", collect.c_str());
std::string customcollect;
for(size_t i = 0; i < SDL_arraysize(obj.customcollect); i++ )
{
customcollect += help.String((int) obj.customcollect[i]) + ",";
}
xml::update_tag(msgs, "customcollect", customcollect.c_str());
//Position
xml::update_tag(msgs, "savex", savex);
xml::update_tag(msgs, "savey", savey);
xml::update_tag(msgs, "saverx", saverx);
xml::update_tag(msgs, "savery", savery);
xml::update_tag(msgs, "savegc", savegc);
xml::update_tag(msgs, "savedir", savedir);
xml::update_tag(msgs, "savepoint", savepoint);
xml::update_tag(msgs, "savecolour", savecolour);
xml::update_tag(msgs, "trinkets", trinkets());
xml::update_tag(msgs, "crewmates", crewmates());
//Special stats
if (music.nicefade)
{
xml::update_tag(msgs, "currentsong", music.nicechange );
}
else
{
xml::update_tag(msgs, "currentsong", music.currentsong);
}
xml::update_tag(msgs, "lang_custom", loc::lang_custom.c_str());
xml::update_tag(msgs, "teleportscript", teleportscript.c_str());
xml::update_tag(msgs, "companion", companion);
xml::update_tag(msgs, "lastsaved", lastsaved);
xml::update_tag(msgs, "supercrewmate", (int) supercrewmate);
xml::update_tag(msgs, "scmprogress", scmprogress);
xml::update_tag(msgs, "frames", frames);
xml::update_tag(msgs, "seconds", seconds);
xml::update_tag(msgs, "minutes", minutes);
xml::update_tag(msgs, "hours", hours);
xml::update_tag(msgs, "deathcounts", deathcounts);
xml::update_tag(msgs, "totalflips", totalflips);
xml::update_tag(msgs, "hardestroom", hardestroom.c_str());
xml::update_tag(msgs, "hardestroomdeaths", hardestroomdeaths);
xml::update_tag(msgs, "hardestroom_x", hardestroom_x);
xml::update_tag(msgs, "hardestroom_y", hardestroom_y);
xml::update_tag(msgs, "hardestroom_specialname", (int) hardestroom_specialname);
xml::update_tag(msgs, "hardestroom_finalstretch", (int) hardestroom_finalstretch);
xml::update_tag(msgs, "showminimap", (int) map.customshowmm);
xml::update_tag(msgs, "disabletemporaryaudiopause", (int) disabletemporaryaudiopause);
xml::update_tag(msgs, "showtrinkets", (int) map.showtrinkets);
if (map.roomnameset)
{
xml::update_tag(msgs, "roomname", map.roomname);
}
else
{
// If there's roomname tags, remove them. There will probably only always be one, but just in case...
tinyxml2::XMLElement* element;
while ((element = msgs->FirstChildElement("roomname")) != NULL)
{
doc.DeleteNode(element);
}
}
std::string legacy_summary = customleveltitle + ", " + timestring();
xml::update_tag(msgs, "summary", legacy_summary.c_str());
if(!FILESYSTEM_saveTiXml2Document(("saves/"+levelfile+".vvv").c_str(), doc))
{
vlog_error("Could Not Save game!");
vlog_error("Failed: %s%s%s", saveFilePath, levelfile.c_str(), ".vvv");
return false;
}
vlog_info("Game saved");
return true;
}
void Game::loadtele(void)
{
tinyxml2::XMLDocument doc;
if (!FILESYSTEM_loadTiXml2Document("saves/tsave.vvv", doc)) return;
readmaingamesave("tsave.vvv", doc);
}
std::string Game::unrescued(void)
{
//Randomly return the name of an unrescued crewmate
//Localization is handled with regular cutscene dialogue
if (fRandom() * 100 > 50)
{
if (!crewstats[5]) return "Victoria";
if (!crewstats[2]) return "Vitellary";
if (!crewstats[4]) return "Verdigris";
if (!crewstats[3]) return "Vermilion";
}
else
{
if (fRandom() * 100 > 50)
{
if (!crewstats[2]) return "Vitellary";
if (!crewstats[4]) return "Verdigris";
if (!crewstats[3]) return "Vermilion";
if (!crewstats[5]) return "Victoria";
}
else
{
if (!crewstats[4]) return "Verdigris";
if (!crewstats[3]) return "Vermilion";
if (!crewstats[5]) return "Victoria";
if (!crewstats[2]) return "Vitellary";
}
}
return "you";
}
void Game::gameclock(void)
{
if (timetrialcountdown > 0)
{
return;
}
frames++;
if (frames >= 30)
{
frames -= 30;
seconds++;
if (seconds >= 60)
{
seconds -= 60;
minutes++;
if (minutes >= 60)
{
minutes -= 60;
hours++;
}
}
}
}
std::string Game::giventimestring( int hrs, int min, int sec )
{
return timetstring(help.hms_to_seconds(hrs, min, sec));
}
std::string Game::timestring(void)
{
return giventimestring(hours, minutes, seconds);
}
std::string Game::resulttimestring(void)
{
//given result time in seconds:
char output[SCREEN_WIDTH_CHARS + 1];
help.format_time(output, sizeof(output), timetrialresulttime, timetrialresultframes, true);
return output;
}
std::string Game::timetstring( int t )
{
//given par time in seconds:
char output[SCREEN_WIDTH_CHARS + 1];
help.format_time(output, sizeof(output), t, -1, true);
return output;
}
void Game::timestringcenti(char* buffer, const size_t buffer_size)
{
help.format_time(buffer, buffer_size, help.hms_to_seconds(hours, minutes, seconds), frames, true);
}
void Game::returnmenu(void)
{
if (menustack.empty())
{
vlog_error("Error: returning to previous menu frame on empty stack!");
return;
}
/* FIXME: Super bad kludge, don't hardcode this! */
if (currentmenuname == Menu::ed_music)
{
music.fadeout();
}
else if (currentmenuname == Menu::gamecompletecontinue
|| currentmenuname == Menu::timetrialcomplete3
|| currentmenuname == Menu::gameover2
|| currentmenuname == Menu::nodeathmodecomplete2)
{
music.play(Music_PRESENTINGVVVVVV);
}
enum Menu::MenuName camefrom = currentmenuname;
MenuStackFrame& frame = menustack[menustack.size()-1];
//Store this in case createmenu() removes the stack frame
int previousoption = frame.option;
createmenu(frame.name, true);
currentmenuoption = previousoption;
//Remove the stackframe now, but createmenu() might have already gotten to it
//if we were returning to the main menu
if (!menustack.empty())
{
menustack.pop_back();
}
/* FIXME: Even more horrible kludge! */
if (camefrom == Menu::timetrialcomplete3)
{
if (can_unlock_ndm())
{
unlock_ndm();
}
}
}
void Game::returntomenu(enum Menu::MenuName t)
{
if (currentmenuname == t)
{
createmenu(t, true);
return;
}
//Unwind the menu stack until we reach our desired menu
int i = menustack.size() - 1;
while (i >= 0)
{
//If we pop it off we can't reference it anymore, so check for it now
bool is_the_menu_we_want = menustack[i].name == t;
returnmenu();
if (is_the_menu_we_want)
{
break;
}
i--;
}
}
void Game::createmenu( enum Menu::MenuName t, bool samemenu/*= false*/ )
{
if (t == Menu::mainmenu && !menutestmode)
{
//Either we've just booted up the game or returned from gamemode
//Whichever it is, we shouldn't have a stack,
//and most likely don't have a current stackframe
menustack.clear();
}
else if (!samemenu)
{
MenuStackFrame frame;
frame.option = currentmenuoption;
frame.name = currentmenuname;
menustack.push_back(frame);
currentmenuoption = 0;
}
currentmenuname = t;
menuyoff = 0;
int maxspacing = 30; // maximum value for menuspacing, can only become lower.
menucountdown = 0;
menuoptions.clear();
switch (t)
{
case Menu::mainmenu:
if (ingame_titlemode)
{
/* We shouldn't be here! */
SDL_assert(0 && "Entering main menu from in-game options!");
break;
}
#if !defined(MAKEANDPLAY)
option(loc::gettext("play"));
#endif
option(loc::gettext("levels"));
option(loc::gettext("options"));
if (loc::show_translator_menu)
{
option(loc::gettext("translator"));
}
option(loc::gettext("credits"));
option(loc::gettext("quit"));
menuyoff = -10;
maxspacing = 15;
break;
case Menu::playerworlds:
option(loc::gettext("play a level"));
option(loc::gettext("level editor"), !editor_disabled);
if (!editor_disabled)
{
option(loc::gettext("open level folder"), FILESYSTEM_openDirectoryEnabled());
option(loc::gettext("show level folder path"));
}
option(loc::gettext("return"));
menuyoff = -40;
maxspacing = 15;
break;
case Menu::confirmshowlevelspath:
option(loc::gettext("no, don't show me"));
option(loc::gettext("yes, reveal the path"));
menuyoff = -10;
break;
case Menu::showlevelspath:
option(loc::gettext("return to levels"));
menuyoff = 60;
break;
case Menu::levellist:
if(cl.ListOfMetaData.size()==0)
{
option(loc::gettext("ok"));
menuyoff = -20;
}
else
{
for(int i=0; i<(int) cl.ListOfMetaData.size(); i++) // FIXME: int/size_t! -flibit
{
if(i>=levelpage*8 && i< (levelpage*8)+8)
{
const std::string filename = cl.ListOfMetaData[i].filename.substr(7);
int score = 0;
if (customlevelstats.count(filename) > 0)
{
score = customlevelstats[filename];
}
const char* prefix;
switch (score)
{
case 0:
{
static const char tmp[] = " ";
prefix = tmp;
break;
}
case 1:
{
static const char tmp[] = " * ";
prefix = tmp;
break;
}
case 3:
{
static const char tmp[] = "** ";
prefix = tmp;
break;
}
default:
SDL_assert(0 && "Unhandled menu text prefix!");
prefix = "";
break;
}
const char* title = cl.ListOfMetaData[i].title.c_str();
if (cl.ListOfMetaData[i].title_is_gettext)
{
title = loc::gettext(title);
}
/* We have to make sure the stars and spaces are consistently on the
* correct side of the title, no matter what bidi characters are in there.
* So just always let the bidi engine handle it, with a few control chars. */
char text[MENU_TEXT_BYTES];
SDL_snprintf(
text, sizeof(text),
"%s%s%s%s%s",
// LRM or RLM depending on UI language, to make the stars aligned to left or right
UTF8_encode(font::is_rtl(PR_FONT_INTERFACE) ? 0x200F : 0x200E).bytes,
prefix,
// FIRST STRONG ISOLATE, to start an isolated block oriented however bidi sees fit
UTF8_encode(0x2068).bytes,
title,
// POP DIRECTIONAL ISOLATE, exit isolated level title
UTF8_encode(0x2069).bytes
);
for (size_t ii = 0; text[ii] != '\0'; ++ii)
{
text[ii] = SDL_tolower(text[ii]);
}
option(
text,
true,
(cl.ListOfMetaData[i].title_is_gettext ? PR_FONT_INTERFACE : PR_FONT_IDX(
cl.ListOfMetaData[i].level_main_font_idx, font::is_rtl(PR_FONT_INTERFACE)
)) | PR_RTL_XFLIP
);
}
}
if (cl.ListOfMetaData.size() > 8)
{
if((size_t) ((levelpage*8)+8) <cl.ListOfMetaData.size())
{
option(loc::gettext("next page"));
}
else
{
option(loc::gettext("first page"));
}
if (levelpage == 0)
{
option(loc::gettext("last page"));
}
else
{
option(loc::gettext("previous page"));
}
}
option(loc::gettext("return"));
menuxoff = 20;
menuyoff = 70-(menuoptions.size()*10);
menuspacing = 5;
return; // skip automatic centering, will turn out bad with levels list
}
break;
case Menu::quickloadlevel:
option(loc::gettext("continue from save"));
option(loc::gettext("start from beginning"));
option(loc::gettext("delete save"));
option(loc::gettext("back to levels"));
menuyoff = -30;
break;
case Menu::deletequicklevel:
option(loc::gettext("no! don't delete"));
option(loc::gettext("yes, delete save"));
menuyoff = 64;
break;
case Menu::youwannaquit:
option(loc::gettext("yes, quit"));
option(loc::gettext("no, return"));
menuyoff = -20;
break;
case Menu::errornostart:
option(loc::gettext("ok"));
menuyoff = -20;
break;
case Menu::gameplayoptions:
#if !defined(MAKEANDPLAY)
if (ingame_titlemode && unlock[Unlock_FLIPMODE])
#endif
{
option(loc::gettext("flip mode"));
}
option(loc::gettext("toggle fps"));
option(loc::gettext("speedrun options"));
option(loc::gettext("advanced options"));
option(loc::gettext("clear main game data"));
option(loc::gettext("clear custom level data"));
option(loc::gettext("return"));
menuyoff = -10;
maxspacing = 15;
break;
case Menu::graphicoptions:
if (!gameScreen.isForcedFullscreen())
{
option(loc::gettext("toggle fullscreen"));
}
option(loc::gettext("scaling mode"));
if (!gameScreen.isForcedFullscreen())
{
option(loc::gettext("resize to nearest"), gameScreen.isWindowed);
}
option(loc::gettext("toggle filter"));
option(loc::gettext("toggle analogue"));
option(loc::gettext("toggle vsync"));
option(loc::gettext("return"));
menuyoff = -10;
maxspacing = 15;
break;
case Menu::ed_settings:
option(loc::gettext("change description"));
option(loc::gettext("edit scripts"));
option(loc::gettext("change music"));
option(loc::gettext("editor ghosts"));
option(loc::gettext("load level"));
option(loc::gettext("save level"));
option(loc::gettext("options"));
option(loc::gettext("quit to main menu"));
menuyoff = -20;
maxspacing = 15;
break;
case Menu::ed_desc:
option(loc::gettext("change name"));
option(loc::gettext("change author"));
option(loc::gettext("change description"));
option(loc::gettext("change website"));
option(loc::gettext("change font"));
option(loc::gettext("return"));
menuyoff = 6;
maxspacing = 15;
break;
case Menu::ed_music:
option(loc::gettext("next song"));
option(loc::gettext("previous song"));
option(loc::gettext("return"));
menuyoff = 16;
maxspacing = 15;
break;
case Menu::ed_quit:
option(loc::gettext("yes, save and quit"));
option(loc::gettext("no, quit without saving"));
option(loc::gettext("return to editor"));
menuyoff = 8;
maxspacing = 15;
break;
case Menu::ed_font:
{
int option_match = -1;
for (uint8_t i = 0; i < font::font_idx_options_n; i++)
{
uint8_t idx = font::font_idx_options[i];
option(font::get_main_font_display_name(idx), true, PR_FONT_IDX(idx, font::is_rtl(PR_FONT_INTERFACE)) | PR_RTL_XFLIP);
if (font::level_font_is_main_idx(idx))
{
option_match = i;
}
}
currentmenuoption = option_match != -1 ? option_match : 0;
maxspacing = 15;
break;
}
case Menu::options:
option(loc::gettext("gameplay"));
option(loc::gettext("graphics"));
option(loc::gettext("audio"));
option(loc::gettext("game pad"));
option(loc::gettext("accessibility"));
option(loc::gettext("language"), !translator_cutscene_test);
option(loc::gettext("return"));
menuyoff = 0;
maxspacing = 15;
break;
case Menu::speedrunneroptions:
option(loc::gettext("glitchrunner mode"));
option(loc::gettext("input delay"));
option(loc::gettext("interact button"));
option(loc::gettext("fake load screen"));
option(loc::gettext("toggle in-game timer"));
option(loc::gettext("english sprites"));
option(loc::gettext("return"));
menuyoff = 0;
maxspacing = 15;
break;
case Menu::setglitchrunner:
{
int i;
option(loc::gettext("none"));
for (i = 1; i < GlitchrunnerNumVersions; ++i)
{
option(loc::gettext(GlitchrunnerMode_enum_to_string((enum GlitchrunnerMode) i)));
}
break;
}
case Menu::advancedoptions:
option(loc::gettext("unfocus pause"));
option(loc::gettext("unfocus audio pause"));
option(loc::gettext("room name background"));
option(loc::gettext("return"));
menuyoff = 0;
maxspacing = 15;
break;
case Menu::audiooptions:
option(loc::gettext("music volume"));
option(loc::gettext("sound volume"));
if (music.mmmmmm)
{
option(loc::gettext("soundtrack"));
}
option(loc::gettext("return"));
menuyoff = 0;
maxspacing = 15;
break;
case Menu::accessibility:
#if !defined(MAKEANDPLAY)
option(loc::gettext("unlock play modes"));
#endif
option(loc::gettext("invincibility"), !ingame_titlemode || !incompetitive());
option(loc::gettext("slowdown"), !ingame_titlemode || !incompetitive());
option(loc::gettext("animated backgrounds"));
option(loc::gettext("screen effects"));
option(loc::gettext("text outline"));
option(loc::gettext("return"));
menuyoff = 0;
maxspacing = 15;
break;
case Menu::controller:
option(loc::gettext("analog stick sensitivity"));
option(loc::gettext("bind flip"));
option(loc::gettext("bind enter"));
option(loc::gettext("bind menu"));
option(loc::gettext("bind restart"));
option(loc::gettext("bind interact"), separate_interact);
option(loc::gettext("return"));
menuyoff = 0;
maxspacing = 10;
break;
case Menu::language:
if (loc::languagelist.empty())
{
option(loc::gettext("ok"));
menuyoff = -20;
}
else
{
for (size_t i = 0; i < loc::languagelist.size(); i++)
{
if (loc::languagelist[i].nativename.empty())
{
option(loc::languagelist[i].code.c_str());
}
else
{
option(
loc::languagelist[i].nativename.c_str(),
true,
PR_FONT_IDX(loc::languagelist[i].font_idx, loc::languagelist[i].rtl)
);
}
}
menuyoff = 70-(menuoptions.size()*10);
maxspacing = 5;
}
break;
case Menu::translator_main:
option(loc::gettext("translator options"));
option(loc::gettext("maintenance"));
option(loc::gettext("open lang folder"), FILESYSTEM_openDirectoryEnabled());
option(loc::gettext("return"));
menuyoff = 0;
break;
case Menu::translator_options:
option(loc::gettext("language statistics"));
option(loc::gettext("translate room names"));
option(loc::gettext("explore game"));
option(loc::gettext("menu test"));
option(loc::gettext("cutscene test"), loc::lang != "en");
option(loc::gettext("limits check"));
option(loc::gettext("return"));
menuyoff = 0;
break;
case Menu::translator_options_limitscheck:
option(loc::gettext("next page"));
option(loc::gettext("return"));
menuyoff = 64;
break;
case Menu::translator_options_stats:
option(loc::gettext("return"));
menuyoff = 64;
break;
case Menu::translator_options_exploregame:
option(loc::gettext("space station 1"));
option(loc::gettext("the laboratory"));
option(loc::gettext("the tower"));
option(loc::gettext("space station 2"));
option(loc::gettext("the warp zone"));
option(loc::gettext("intermission 1"));
option(loc::gettext("intermission 2"));
option(loc::gettext("the final level"));
option(loc::gettext("return"));
menuyoff = -20;
break;
case Menu::translator_options_cutscenetest:
for (
size_t i = (cutscenetest_menu_page*14);
i < (cutscenetest_menu_page*14)+14 && i < loc::testable_script_ids.size();
i++
)
{
option(loc::testable_script_ids[i].c_str());
}
if((cutscenetest_menu_page*14)+14 < loc::testable_script_ids.size())
{
option(loc::gettext("next page"));
}
else
{
option(loc::gettext("first page"));
}
if (cutscenetest_menu_page == 0)
{
option(loc::gettext("last page"));
}
else
{
option(loc::gettext("previous page"));
}
option(loc::gettext("from clipboard"));
option(loc::gettext("return"));
menuxoff = 20;
menuyoff = 55-(menuoptions.size()*10);
menuspacing = 5;
return; // skip automatic centering, will turn out bad with scripts list
case Menu::translator_maintenance:
option(loc::gettext("sync language files"));
option(loc::gettext("global statistics"), false);
option(loc::gettext("global limits check"));
option(loc::gettext("return"));
menuyoff = 0;
break;
case Menu::translator_maintenance_sync:
option(loc::gettext("sync"));
option(loc::gettext("return"));
menuyoff = 64;
break;
case Menu::translator_error_setlangwritedir:
option(loc::gettext("ok"));
menuyoff = 10;
break;
case Menu::cleardatamenu:
case Menu::clearcustomdatamenu:
option(loc::gettext("no! don't delete"));
option(loc::gettext("yes, delete everything"));
menuyoff = 64;
break;
case Menu::setinvincibility:
option(loc::gettext("no, return to options"));
option(loc::gettext("yes, enable"));
menuyoff = 64;
break;
case Menu::setslowdown:
option(loc::gettext("normal speed"));
option(loc::gettext("80% speed"));
option(loc::gettext("60% speed"));
option(loc::gettext("40% speed"));
menuyoff = 16;
break;
case Menu::unlockmenu:
option(loc::gettext("unlock time trials"));
option(loc::gettext("unlock intermissions"), !unlock[Unlock_INTERMISSION_REPLAYS]);
option(loc::gettext("unlock no death mode"), !unlock[Unlock_NODEATHMODE]);
option(loc::gettext("unlock flip mode"), !unlock[Unlock_FLIPMODE]);
option(loc::gettext("unlock ship jukebox"), (stat_trinkets<20));
option(loc::gettext("unlock secret lab"), !unlock[Unlock_SECRETLAB]);
option(loc::gettext("return"));
menuyoff = -20;
break;
case Menu::credits:
option(loc::gettext("next page"));
option(loc::gettext("last page"));
option(loc::gettext("return"));
menuyoff = 64;
break;
case Menu::credits2:
case Menu::credits25:
case Menu::credits3:
case Menu::credits4:
case Menu::credits5:
case Menu::credits_localisations_implementation:
case Menu::credits_localisations_translations:
option(loc::gettext("next page"));
option(loc::gettext("previous page"));
option(loc::gettext("return"));
menuyoff = 64;
break;
case Menu::credits6:
option(loc::gettext("first page"));
option(loc::gettext("previous page"));
option(loc::gettext("return"));
menuyoff = 64;
break;
case Menu::play:
{
//Ok, here's where the unlock stuff comes into it:
//First up, time trials:
int temp = 0;
if (unlock[Unlock_SPACESTATION1_COMPLETE]
&& stat_trinkets >= 3
&& !unlocknotify[Unlock_TIMETRIAL_SPACESTATION1])
{
temp++;
}
if (unlock[Unlock_LABORATORY_COMPLETE]
&& stat_trinkets >= 6
&& !unlocknotify[Unlock_TIMETRIAL_LABORATORY])
{
temp++;
}
if (unlock[Unlock_TOWER_COMPLETE]
&& stat_trinkets >= 9
&& !unlocknotify[Unlock_TIMETRIAL_TOWER])
{
temp++;
}
if (unlock[Unlock_SPACESTATION2_COMPLETE]
&& stat_trinkets >= 12
&& !unlocknotify[Unlock_TIMETRIAL_SPACESTATION2])
{
temp++;
}
if (unlock[Unlock_WARPZONE_COMPLETE]
&& stat_trinkets >= 15
&& !unlocknotify[Unlock_TIMETRIAL_WARPZONE])
{
temp++;
}
if (unlock[UnlockTrophy_GAME_COMPLETE]
&& stat_trinkets >= 18
&& !unlocknotify[Unlock_TIMETRIAL_FINALLEVEL])
{
temp++;
}
if (temp > 0)
{
//you've unlocked a time trial!
if (unlock[Unlock_SPACESTATION1_COMPLETE] && stat_trinkets >= 3)
{
unlocknotify[Unlock_TIMETRIAL_SPACESTATION1] = true;
unlock[Unlock_TIMETRIAL_SPACESTATION1] = true;
}
if (unlock[Unlock_LABORATORY_COMPLETE] && stat_trinkets >= 6)
{
unlocknotify[Unlock_TIMETRIAL_LABORATORY] = true;
unlock[Unlock_TIMETRIAL_LABORATORY] = true;
}
if (unlock[Unlock_TOWER_COMPLETE] && stat_trinkets >= 9)
{
unlocknotify[Unlock_TIMETRIAL_TOWER] = true;
unlock[Unlock_TIMETRIAL_TOWER] = true;
}
if (unlock[Unlock_SPACESTATION2_COMPLETE] && stat_trinkets >= 12)
{
unlocknotify[Unlock_TIMETRIAL_SPACESTATION2] = true;
unlock[Unlock_TIMETRIAL_SPACESTATION2] = true;
}
if (unlock[Unlock_WARPZONE_COMPLETE] && stat_trinkets >= 15)
{
unlocknotify[Unlock_TIMETRIAL_WARPZONE] = true;
unlock[Unlock_TIMETRIAL_WARPZONE] = true;
}
if (unlock[UnlockTrophy_GAME_COMPLETE] && stat_trinkets >= 18)
{
unlocknotify[Unlock_TIMETRIAL_FINALLEVEL] = true;
unlock[Unlock_TIMETRIAL_FINALLEVEL] = true;
}
if (temp == 1)
{
createmenu(Menu::unlocktimetrial, true);
savestatsandsettings();
}
else if (temp > 1)
{
createmenu(Menu::unlocktimetrials, true);
savestatsandsettings();
}
}
else
{
//Alright, we haven't unlocked any time trials. How about no death mode?
if (can_unlock_ndm())
{
unlock_ndm();
}
//Alright then! Flip mode?
else if (unlock[UnlockTrophy_GAME_COMPLETE]
&& !unlocknotify[Unlock_FLIPMODE])
{
unlock[Unlock_FLIPMODE] = true;
unlocknotify[Unlock_FLIPMODE] = true;
createmenu(Menu::unlockflipmode, true);
savestatsandsettings();
}
//What about the intermission levels?
else if (unlock[Unlock_INTERMISSION2_COMPLETE]
&& !unlocknotify[Unlock_INTERMISSION_REPLAYS])
{
unlock[Unlock_INTERMISSION_REPLAYS] = true;
unlocknotify[Unlock_INTERMISSION_REPLAYS] = true;
createmenu(Menu::unlockintermission, true);
savestatsandsettings();
}
else
{
if (save_exists())
{
option(loc::gettext("continue"));
}
else
{
option(loc::gettext("new game"));
}
//ok, secret lab! no notification, but test:
if (unlock[Unlock_SECRETLAB])
{
option(loc::gettext("secret lab"));
}
option(loc::gettext("play modes"));
if (save_exists())
{
option(loc::gettext("new game"));
}
option(loc::gettext("return"));
if (unlock[Unlock_SECRETLAB])
{
menuyoff = -30;
}
else
{
menuyoff = -40;
}
}
}
break;
}
case Menu::unlocktimetrial:
case Menu::unlocktimetrials:
case Menu::unlocknodeathmode:
case Menu::unlockintermission:
case Menu::unlockflipmode:
option(loc::gettext("proceed"));
menuyoff = 70;
break;
case Menu::newgamewarning:
option(loc::gettext("start new game"));
option(loc::gettext("return"));
menuyoff = 64;
break;
case Menu::playmodes:
option(loc::gettext("time trials"), !nocompetitive_unless_translator());
option(loc::gettext("intermissions"), unlock[Unlock_INTERMISSION_REPLAYS]);
option(loc::gettext("no death mode"), unlock[Unlock_NODEATHMODE] && !nocompetitive());
option(loc::gettext("flip mode"), unlock[Unlock_FLIPMODE]);
option(loc::gettext("return"));
menuyoff = 8;
maxspacing = 20;
break;
case Menu::intermissionmenu:
option(loc::gettext("play intermission 1"));
option(loc::gettext("play intermission 2"));
option(loc::gettext("return"));
menuyoff = -35;
break;
case Menu::playint1:
start_translator_exploring = false;
option(loc::gettext_case("Vitellary", 1));
option(loc::gettext_case("Vermilion", 1));
option(loc::gettext_case("Verdigris", 1));
option(loc::gettext_case("Victoria", 1));
option(loc::gettext("return"));
menuyoff = 10;
break;
case Menu::playint2:
start_translator_exploring = false;
option(loc::gettext_case("Vitellary", 1));
option(loc::gettext_case("Vermilion", 1));
option(loc::gettext_case("Verdigris", 1));
option(loc::gettext_case("Victoria", 1));
option(loc::gettext("return"));
menuyoff = 10;
break;
case Menu::continuemenu:
map.settowercolour(3);
option(loc::gettext("continue from teleporter"));
option(loc::gettext("continue from quicksave"));
option(loc::gettext("return"));
menuyoff = 20;
break;
case Menu::startnodeathmode:
option(loc::gettext("disable cutscenes"));
option(loc::gettext("enable cutscenes"));
option(loc::gettext("return"));
menuyoff = 40;
break;
case Menu::gameover:
menucountdown = 120;
menudest=Menu::gameover2;
break;
case Menu::gameover2:
option(loc::gettext("return to play menu"));
menuyoff = 80;
break;
case Menu::unlockmenutrials:
option(loc::gettext("space station 1"), !unlock[Unlock_TIMETRIAL_SPACESTATION1]);
option(loc::gettext("the laboratory"), !unlock[Unlock_TIMETRIAL_LABORATORY]);
option(loc::gettext("the tower"), !unlock[Unlock_TIMETRIAL_TOWER]);
option(loc::gettext("space station 2"), !unlock[Unlock_TIMETRIAL_SPACESTATION2]);
option(loc::gettext("the warp zone"), !unlock[Unlock_TIMETRIAL_WARPZONE]);
option(loc::gettext("the final level"), !unlock[Unlock_TIMETRIAL_FINALLEVEL]);
option(loc::gettext("return"));
menuyoff = 0;
break;
case Menu::timetrials:
option(loc::gettext(unlock[Unlock_TIMETRIAL_SPACESTATION1] ? "space station 1" : "???"),
unlock[Unlock_TIMETRIAL_SPACESTATION1]);
option(loc::gettext(unlock[Unlock_TIMETRIAL_LABORATORY] ? "the laboratory" : "???"),
unlock[Unlock_TIMETRIAL_LABORATORY]);
option(loc::gettext(unlock[Unlock_TIMETRIAL_TOWER] ? "the tower" : "???"),
unlock[Unlock_TIMETRIAL_TOWER]);
option(loc::gettext(unlock[Unlock_TIMETRIAL_SPACESTATION2] ? "space station 2" : "???"),
unlock[Unlock_TIMETRIAL_SPACESTATION2]);
option(loc::gettext(unlock[Unlock_TIMETRIAL_WARPZONE] ? "the warp zone" : "???"),
unlock[Unlock_TIMETRIAL_WARPZONE]);
option(loc::gettext(unlock[Unlock_TIMETRIAL_FINALLEVEL] ? "the final level" : "???"),
unlock[Unlock_TIMETRIAL_FINALLEVEL]);
option(loc::gettext("return"));
menuyoff = 0;
maxspacing = 15;
break;
case Menu::nodeathmodecomplete:
menucountdown = 90;
menudest = Menu::nodeathmodecomplete2;
break;
case Menu::nodeathmodecomplete2:
option(loc::gettext("return to play menu"));
menuyoff = 70;
break;
case Menu::timetrialcomplete:
menucountdown = 90;
menudest=Menu::timetrialcomplete2;
break;
case Menu::timetrialcomplete2:
menucountdown = 60;
menudest=Menu::timetrialcomplete3;
break;
case Menu::timetrialcomplete3:
option(loc::gettext("return to play menu"));
option(loc::gettext("try again"));
menuyoff = 70;
break;
case Menu::gamecompletecontinue:
option(loc::gettext("return to play menu"));
menuyoff = 70;
break;
case Menu::errorsavingsettings:
option(loc::gettext("ok"));
option(loc::gettext("silence"));
menuyoff = 10;
break;
case Menu::errorloadinglevel:
case Menu::warninglevellist:
option(loc::gettext("ok"));
menuyoff = 50;
break;
}
// Automatically center the menu. We must check the width of the menu with the initial horizontal spacing.
// If it's too wide, reduce the horizontal spacing by 5 and retry.
// Try to limit the menu width to 272 pixels: 320 minus 16*2 for square brackets, minus 8*2 padding.
// The square brackets fall outside the menu width (i.e. selected menu options are printed 16 pixels to the left)
bool done_once = false;
int menuwidth = 0;
for (; !done_once || (menuwidth > 272 && menuspacing > 0); maxspacing -= 5)
{
done_once = true;
menuspacing = maxspacing;
menuwidth = 0;
for (size_t i = 0; i < menuoptions.size(); i++)
{
int width = i*menuspacing + font::len(menuoptions[i].print_flags, menuoptions[i].text);
if (width > menuwidth)
menuwidth = width;
}
}
menuxoff = (320-menuwidth)/2;
}
bool Game::can_unlock_ndm(void)
{
int temp = 0;
if (bestrank[TimeTrial_SPACESTATION1] >= 2) temp++;
if (bestrank[TimeTrial_LABORATORY] >= 2) temp++;
if (bestrank[TimeTrial_TOWER] >= 2) temp++;
if (bestrank[TimeTrial_SPACESTATION2] >= 2) temp++;
if (bestrank[TimeTrial_WARPZONE] >= 2) temp++;
if (bestrank[TimeTrial_FINALLEVEL] >= 2) temp++;
return temp >= 4 && !unlocknotify[Unlock_NODEATHMODE];
}
void Game::unlock_ndm(void)
{
unlocknotify[Unlock_NODEATHMODE] = true;
unlock[Unlock_NODEATHMODE] = true;
createmenu(Menu::unlocknodeathmode, true);
savestatsandsettings();
}
void Game::deletequick(void)
{
if (inspecial() || map.custommode)
{
return;
}
if (!FILESYSTEM_delete("saves/qsave.vvv"))
{
vlog_error("Error deleting saves/qsave.vvv");
}
else
{
SDL_zero(last_quicksave);
}
}
void Game::deletetele(void)
{
if (inspecial() || map.custommode)
{
return;
}
if (!FILESYSTEM_delete("saves/tsave.vvv"))
{
vlog_error("Error deleting saves/tsave.vvv");
}
else
{
SDL_zero(last_telesave);
}
}
void Game::customdeletequick(const std::string& file)
{
const std::string path = "saves/" + file.substr(7) + ".vvv";
if (!FILESYSTEM_delete(path.c_str()))
{
vlog_error("Error deleting %s", path.c_str());
}
}
void Game::swnpenalty(void)
{
//set the SWN clock back to the closest 5 second interval
if (swntimer <= 150)
{
swntimer += 8;
if (swntimer > 150) swntimer = 150;
}
else if (swntimer <= 300)
{
swntimer += 8;
if (swntimer > 300) swntimer = 300;
}
else if (swntimer <= 450)
{
swntimer += 8;
if (swntimer > 450) swntimer = 450;
}
else if (swntimer <= 600)
{
swntimer += 8;
if (swntimer > 600) swntimer = 600;
}
else if (swntimer <= 750)
{
swntimer += 8;
if (swntimer > 750) swntimer = 750;
}
else if (swntimer <= 900)
{
swntimer += 8;
if (swntimer > 900) swntimer = 900;
}
else if (swntimer <= 1050)
{
swntimer += 8;
if (swntimer > 1050) swntimer = 1050;
}
else if (swntimer <= 1200)
{
swntimer += 8;
if (swntimer > 1200) swntimer = 1200;
}
else if (swntimer <= 1350)
{
swntimer += 8;
if (swntimer > 1350) swntimer = 1350;
}
else if (swntimer <= 1500)
{
swntimer += 8;
if (swntimer > 1500) swntimer = 1500;
}
else if (swntimer <= 1650)
{
swntimer += 8;
if (swntimer > 1650) swntimer = 1650;
}
else if (swntimer <= 1800)
{
swntimer += 8;
if (swntimer > 1800) swntimer = 1800;
}
else if (swntimer <= 2100)
{
swntimer += 8;
if (swntimer > 2100) swntimer = 2100;
}
else if (swntimer <= 2400)
{
swntimer += 8;
if (swntimer > 2400) swntimer = 2400;
}
}
int Game::crewrescued(void)
{
int temp = 0;
for (size_t i = 0; i < SDL_arraysize(crewstats); i++)
{
if (crewstats[i])
{
temp++;
}
}
return temp;
}
void Game::resetgameclock(void)
{
frames = 0;
seconds = 0;
minutes = 0;
hours = 0;
}
int Game::trinkets(void)
{
int temp = 0;
for (size_t i = 0; i < SDL_arraysize(obj.collect); i++)
{
if (obj.collect[i])
{
temp++;
}
}
return temp;
}
int Game::crewmates(void)
{
int temp = 0;
for (size_t i = 0; i < SDL_arraysize(obj.customcollect); i++)
{
if (obj.customcollect[i])
{
temp++;
}
}
return temp;
}
bool Game::anything_unlocked(void)
{
for (size_t i = 0; i < SDL_arraysize(unlock); i++)
{
if (unlock[i] &&
(i == 8 // Secret Lab
|| (i >= 9 && i <= 14) // any Time Trial
|| i == 16 // Intermission replays
|| i == 17 // No Death Mode
|| i == 18)) // Flip Mode
{
return true;
}
}
return false;
}
bool Game::save_exists(void)
{
return last_telesave.exists || last_quicksave.exists;
}
static void hardreset(void)
{
script.hardreset();
}
static void returntoeditor_callback(void)
{
extern Game game;
game.returntoeditor();
ed.show_note(loc::gettext("Level quits to menu"));
}
void Game::quittomenu(void)
{
if (gamestate != EDITORMODE && map.custommode && !map.custommodeforreal)
{
/* We are playtesting! Go back to the editor
* instead of losing unsaved changes. */
/* This needs to be deferred, otherwise some state would persist. */
DEFER_CALLBACK(returntoeditor_callback);
return;
}
gamestate = TITLEMODE;
graphics.fademode = FADE_START_FADEIN;
FILESYSTEM_unmountAssets();
loc::unloadtext_custom();
font::unload_custom();
cliplaytest = false;
graphics.titlebg.tdrawback = true;
graphics.flipmode = false;
//Don't be stuck on the summary screen,
//or "who do you want to play the level with?"
//or "do you want cutscenes?"
//or the confirm-load-quicksave menu
if (translator_cutscene_test)
{
returntomenu(Menu::translator_options_cutscenetest);
}
else if (translator_exploring)
{
returntomenu(Menu::translator_options_exploregame);
}
else if (intimetrial)
{
returntomenu(Menu::timetrials);
}
else if (inintermission)
{
returntomenu(Menu::intermissionmenu);
}
else if (nodeathmode)
{
returntomenu(Menu::playmodes);
}
else if (map.custommode)
{
if (map.custommodeforreal)
{
returntomenu(Menu::levellist);
}
else
{
//Returning from editor
editor_disabled = !BUTTONGLYPHS_keyboard_is_available();
returntomenu(Menu::playerworlds);
}
}
else if (save_exists() || anything_unlocked())
{
returntomenu(Menu::play);
if (!insecretlab)
{
//Select "continue"
currentmenuoption = 0;
}
}
else
{
createmenu(Menu::mainmenu);
}
/* We might not be at the end of the frame yet.
* If we hardreset() now, some state might still persist. */
DEFER_CALLBACK(hardreset);
}
void Game::returntolab(void)
{
gamestate = GAMEMODE;
graphics.fademode = FADE_START_FADEIN;
map.gotoroom(119, 107);
int player = obj.getplayer();
if (INBOUNDS_VEC(player, obj.entities))
{
obj.entities[player].xp = 132;
obj.entities[player].yp = 137;
}
gravitycontrol = 0;
savepoint = 0;
saverx = 119;
savery = 107;
savex = 132;
savey = 137;
savegc = 0;
if (INBOUNDS_VEC(player, obj.entities))
{
savedir = obj.entities[player].dir;
}
music.play(Music_PIPEDREAM);
}
static void resetbg(void)
{
graphics.backgrounddrawn = false;
}
void Game::returntoeditor(void)
{
gamestate = EDITORMODE;
graphics.textboxes.clear();
hascontrol = true;
advancetext = false;
completestop = false;
setstate(0);
graphics.showcutscenebars = false;
graphics.fademode = FADE_NONE;
ed.keydelay = 6;
ed.settingskey = true;
ed.old_note_timer = 0;
ed.note_timer = 0;
ed.roomnamehide = 0;
// Might've been changed in a script
font::set_level_font(cl.level_font_name.c_str());
DEFER_CALLBACK(resetbg);
music.fadeout();
//If warpdir() is used during playtesting, we need to set it back after!
for (int j = 0; j < cl.maxheight; j++)
{
for (int i = 0; i < cl.maxwidth; i++)
{
cl.roomproperties[i+(j*cl.maxwidth)].warpdir=ed.kludgewarpdir[i+(j*cl.maxwidth)];
}
}
graphics.titlebg.scrolldir = 0;
graphics.backgrounddrawn = false;
graphics.foregrounddrawn = false;
}
static void returntoingametemp(void)
{
extern Game game;
game.returntomenu(game.kludge_ingametemp);
}
static void returntoedsettings(void)
{
extern Game game;
game.returntomenu(Menu::ed_settings);
}
static void nextbgcolor(void)
{
map.nexttowercolour();
}
static void setfademode(void)
{
graphics.fademode = graphics.ingame_fademode;
}
static void setflipmode(void)
{
graphics.flipmode = graphics.setflipmode;
}
void Game::returntoingame(void)
{
ingame_titlemode = false;
mapheld = true;
if (ingame_editormode)
{
ingame_editormode = false;
DEFER_CALLBACK(returntoedsettings);
gamestate = EDITORMODE;
ed.settingskey = true;
}
else
{
DEFER_CALLBACK(returntoingametemp);
gamestate = MAPMODE;
DEFER_CALLBACK(setflipmode);
DEFER_CALLBACK(setfademode);
if (!map.custommode && !graphics.setflipmode)
{
obj.flags[73] = true;
}
}
DEFER_CALLBACK(nextbgcolor);
}
void Game::unlockAchievement(const char* name)
{
#ifdef MAKEANDPLAY
UNUSED(name);
#else
if (map.custommode)
{
return;
}
vlog_debug("Achievement \"%s\" unlocked.", name);
NETWORK_unlockAchievement(name);
#endif
}
void Game::mapmenuchange(const enum GameGamestate newgamestate, const bool user_initiated)
{
if (user_initiated && graphics.resumegamemode)
{
return;
}
prevgamestate = gamestate;
gamestate = newgamestate;
graphics.resumegamemode = false;
mapheld = true;
gameScreen.recacheTextures();
if (prevgamestate == GAMEMODE)
{
graphics.menuoffset = 240;
}
else
{
graphics.menuoffset = 0;
}
graphics.oldmenuoffset = graphics.menuoffset;
}
void Game::copyndmresults(void)
{
ndmresultcrewrescued = crewrescued();
ndmresulttrinkets = trinkets();
ndmresulthardestroom = hardestroom;
ndmresulthardestroom_x = hardestroom_x;
ndmresulthardestroom_y = hardestroom_y;
ndmresulthardestroom_specialname = hardestroom_specialname;
SDL_memcpy(ndmresultcrewstats, crewstats, sizeof(ndmresultcrewstats));
}
static inline int get_framerate(const int slowdown)
{
switch (slowdown)
{
case 30:
return 34;
case 24:
return 41;
case 18:
return 55;
case 12:
return 83;
}
return 34;
}
int Game::get_timestep(void)
{
if ((gamestate == GAMEMODE || (gamestate == TELEPORTERMODE && !useteleporter)) &&
level_debugger::is_active() &&
!level_debugger::is_pausing() &&
key.isDown(SDLK_f))
{
return 1;
}
switch (gamestate)
{
case GAMEMODE:
return get_framerate(slowdown);
default:
return 34;
}
}
bool Game::physics_frozen(void)
{
return roomname_translator::is_pausing() || level_debugger::is_pausing();
}
bool Game::incompetitive(void)
{
return (
!map.custommode
&& swnmode
&& (swngame == SWN_SUPERGRAVITRON ||
swngame == SWN_START_SUPERGRAVITRON_STEP_1 ||
swngame == SWN_START_SUPERGRAVITRON_STEP_2)
)
|| intimetrial
|| nodeathmode;
}
bool Game::nocompetitive(void)
{
return slowdown < 30 || map.invincibility;
}
bool Game::nocompetitive_unless_translator(void)
{
return slowdown < 30 || (map.invincibility && !roomname_translator::enabled);
}
void Game::sabotage_time_trial(void)
{
timetrialcheater = true;
hours++;
deathcounts += 100;
timetrialparlost = true;
}
bool Game::isingamecompletescreen(void)
{
return (state >= 3501 && state <= 3518) || (state >= 3520 && state <= 3522);
}