scummvm/engines/lastexpress/debug.cpp

1174 lines
29 KiB
C++

/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
*/
#include "lastexpress/debug.h"
// Data
#include "lastexpress/data/animation.h"
#include "lastexpress/data/background.h"
#include "lastexpress/data/cursor.h"
#include "lastexpress/data/scene.h"
#include "lastexpress/data/sequence.h"
#include "lastexpress/data/subtitle.h"
#include "lastexpress/fight/fight.h"
#include "lastexpress/game/action.h"
#include "lastexpress/game/beetle.h"
#include "lastexpress/game/inventory.h"
#include "lastexpress/game/logic.h"
#include "lastexpress/game/object.h"
#include "lastexpress/game/savegame.h"
#include "lastexpress/game/savepoint.h"
#include "lastexpress/game/scenes.h"
#include "lastexpress/game/state.h"
#include "lastexpress/sound/queue.h"
#include "lastexpress/graphics.h"
#include "lastexpress/lastexpress.h"
#include "lastexpress/resource.h"
#include "common/debug-channels.h"
#include "common/md5.h"
namespace LastExpress {
Debugger::Debugger(LastExpressEngine *engine) : _engine(engine), _command(NULL), _numParams(0), _commandParams(NULL) {
//////////////////////////////////////////////////////////////////////////
// Register the debugger commands
// General
DCmd_Register("help", WRAP_METHOD(Debugger, cmdHelp));
// Data
DCmd_Register("ls", WRAP_METHOD(Debugger, cmdListFiles));
DCmd_Register("dump", WRAP_METHOD(Debugger, cmdDumpFiles));
DCmd_Register("showframe", WRAP_METHOD(Debugger, cmdShowFrame));
DCmd_Register("showbg", WRAP_METHOD(Debugger, cmdShowBg));
DCmd_Register("playseq", WRAP_METHOD(Debugger, cmdPlaySeq));
DCmd_Register("playsnd", WRAP_METHOD(Debugger, cmdPlaySnd));
DCmd_Register("playsbe", WRAP_METHOD(Debugger, cmdPlaySbe));
DCmd_Register("playnis", WRAP_METHOD(Debugger, cmdPlayNis));
// Scene & interaction
DCmd_Register("loadscene", WRAP_METHOD(Debugger, cmdLoadScene));
DCmd_Register("fight", WRAP_METHOD(Debugger, cmdFight));
DCmd_Register("beetle", WRAP_METHOD(Debugger, cmdBeetle));
// Game
DCmd_Register("delta", WRAP_METHOD(Debugger, cmdTimeDelta));
DCmd_Register("time", WRAP_METHOD(Debugger, cmdTime));
DCmd_Register("show", WRAP_METHOD(Debugger, cmdShow));
DCmd_Register("entity", WRAP_METHOD(Debugger, cmdEntity));
// Misc
DCmd_Register("loadgame", WRAP_METHOD(Debugger, cmdLoadGame));
DCmd_Register("chapter", WRAP_METHOD(Debugger, cmdSwitchChapter));
DCmd_Register("clear", WRAP_METHOD(Debugger, cmdClear));
resetCommand();
_soundStream = new StreamedSound();
}
Debugger::~Debugger() {
DebugMan.clearAllDebugChannels();
SAFE_DELETE(_soundStream);
resetCommand();
_command = NULL;
_commandParams = NULL;
// Zero passed pointers
_engine = NULL;
}
//////////////////////////////////////////////////////////////////////////
// Helper functions
//////////////////////////////////////////////////////////////////////////
bool Debugger::hasCommand() const {
return (_numParams != 0);
}
void Debugger::resetCommand() {
SAFE_DELETE(_command);
if (_commandParams)
for (int i = 0; i < _numParams; i++)
free(_commandParams[i]);
free(_commandParams);
_commandParams = NULL;
_numParams = 0;
}
int Debugger::getNumber(const char *arg) const {
return strtol(arg, (char **)NULL, 0);
}
void Debugger::copyCommand(int argc, const char **argv) {
_commandParams = (char **)malloc(sizeof(char *) * (uint)argc);
if (!_commandParams)
return;
_numParams = argc;
for (int i = 0; i < _numParams; i++) {
_commandParams[i] = (char *)malloc(strlen(argv[i]) + 1);
memset(_commandParams[i], 0, strlen(argv[i]) + 1);
strcpy(_commandParams[i], argv[i]);
}
// Exit the debugger!
Cmd_Exit(0, 0);
}
void Debugger::callCommand() {
if (_command)
(*_command)(_numParams, const_cast<const char **>(_commandParams));
}
void Debugger::loadArchive(ArchiveIndex index) const {
_engine->getResourceManager()->loadArchive(index);
getScenes()->loadSceneDataFile(index);
}
// Restore loaded archive
void Debugger::restoreArchive() const {
ArchiveIndex index = kArchiveCd1;
switch (getProgress().chapter) {
default:
case kChapter1:
index = kArchiveCd1;
break;
case kChapter2:
case kChapter3:
index = kArchiveCd2;
break;
case kChapter4:
case kChapter5:
index = kArchiveCd3;
break;
}
_engine->getResourceManager()->loadArchive(index);
getScenes()->loadSceneDataFile(index);
}
//////////////////////////////////////////////////////////////////////////
// Debugger commands
//////////////////////////////////////////////////////////////////////////
bool Debugger::cmdHelp(int, const char **) {
DebugPrintf("Debug flags\n");
DebugPrintf("-----------\n");
DebugPrintf(" debugflag_list - Lists the available debug flags and their status\n");
DebugPrintf(" debugflag_enable - Enables a debug flag\n");
DebugPrintf(" debugflag_disable - Disables a debug flag\n");
DebugPrintf("\n");
DebugPrintf("Commands\n");
DebugPrintf("--------\n");
DebugPrintf(" ls - list files in the archive\n");
DebugPrintf(" dump - dump a list of files in all archives\n");
DebugPrintf("\n");
DebugPrintf(" showframe - show a frame from a sequence\n");
DebugPrintf(" showbg - show a background\n");
DebugPrintf(" playseq - play a sequence\n");
DebugPrintf(" playsnd - play a sound\n");
DebugPrintf(" playsbe - play a subtitle\n");
DebugPrintf(" playnis - play an animation\n");
DebugPrintf("\n");
DebugPrintf(" loadscene - load a scene\n");
DebugPrintf(" fight - start a fight\n");
DebugPrintf(" beetle - start the beetle game\n");
DebugPrintf("\n");
DebugPrintf(" delta - Adjust the time delta\n");
DebugPrintf(" show - show game data\n");
DebugPrintf(" entity - show entity data\n");
DebugPrintf("\n");
DebugPrintf(" loadgame - load a saved game\n");
DebugPrintf(" chapter - switch to a specific chapter\n");
DebugPrintf(" clear - clear the screen\n");
DebugPrintf("\n");
return true;
}
/**
* Command: list files in archive
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdListFiles(int argc, const char **argv) {
if (argc == 2 || argc == 3) {
Common::String filter(const_cast<char *>(argv[1]));
// Load the proper archive
if (argc == 3)
loadArchive((ArchiveIndex)getNumber(argv[2]));
Common::ArchiveMemberList list;
int count = _engine->getResourceManager()->listMatchingMembers(list, filter);
DebugPrintf("Number of matches: %d\n", count);
for (Common::ArchiveMemberList::iterator it = list.begin(); it != list.end(); ++it)
DebugPrintf(" %s\n", (*it)->getName().c_str());
// Restore archive
if (argc == 3)
restoreArchive();
} else {
DebugPrintf("Syntax: ls <filter> (use * for all) (<cd number>)\n");
}
return true;
}
/**
* Command: Dump the list of files in the archive
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdDumpFiles(int argc, const char **) {
#define OUTPUT_ARCHIVE_FILES(name, filename) { \
_engine->getResourceManager()->reset(); \
_engine->getResourceManager()->loadArchive(filename); \
Common::ArchiveMemberList list; \
int count = _engine->getResourceManager()->listMatchingMembers(list, "*"); \
debugC(1, kLastExpressDebugResource, "\n\n--------------------------------------------------------------------\n"); \
debugC(1, kLastExpressDebugResource, "-- " #name " (%d files)\n", count); \
debugC(1, kLastExpressDebugResource, "--------------------------------------------------------------------\n\n"); \
debugC(1, kLastExpressDebugResource, "Filename,Size,MD5\n"); \
for (Common::ArchiveMemberList::iterator it = list.begin(); it != list.end(); ++it) { \
Common::SeekableReadStream *stream = getArchive((*it)->getName()); \
if (!stream) { \
DebugPrintf("ERROR: Cannot create stream for file: %s\n", (*it)->getName().c_str()); \
restoreArchive(); \
return true; \
} \
Common::String md5str = Common::computeStreamMD5AsString(*stream); \
debugC(1, kLastExpressDebugResource, "%s, %d, %s", (*it)->getName().c_str(), stream->size(), md5str.c_str()); \
delete stream; \
} \
}
if (argc == 1) {
// For each archive file, dump the list of files
if (_engine->isDemo()) {
OUTPUT_ARCHIVE_FILES("DEMO", "DEMO.HPF");
} else {
OUTPUT_ARCHIVE_FILES("HD", "HD.HPF");
OUTPUT_ARCHIVE_FILES("CD 1", "CD1.HPF");
OUTPUT_ARCHIVE_FILES("CD 2", "CD2.HPF");
OUTPUT_ARCHIVE_FILES("CD 3", "CD3.HPF");
}
// Restore current loaded archive
restoreArchive();
} else {
DebugPrintf("Syntax: dump");
}
return true;
}
/**
* Command: Shows a frame
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdShowFrame(int argc, const char **argv) {
if (argc == 3 || argc == 4) {
Common::String filename(const_cast<char *>(argv[1]));
filename += ".seq";
if (argc == 4)
loadArchive((ArchiveIndex)getNumber(argv[3]));
if (!_engine->getResourceManager()->hasFile(filename)) {
DebugPrintf("Cannot find file: %s\n", filename.c_str());
return true;
}
// Store command
if (!hasCommand()) {
_command = WRAP_METHOD(Debugger, cmdShowFrame);
copyCommand(argc, argv);
return Cmd_Exit(0, 0);
} else {
Sequence sequence(filename);
if (sequence.load(getArchive(filename))) {
_engine->getCursor()->show(false);
clearBg(GraphicsManager::kBackgroundOverlay);
AnimFrame *frame = sequence.getFrame((uint16)getNumber(argv[2]));
if (!frame) {
DebugPrintf("Invalid frame index '%s'\n", argv[2]);
resetCommand();
return true;
}
_engine->getGraphicsManager()->draw(frame, GraphicsManager::kBackgroundOverlay);
delete frame;
askForRedraw();
redrawScreen();
_engine->_system->delayMillis(1000);
_engine->getCursor()->show(true);
}
resetCommand();
if (argc == 4)
restoreArchive();
}
} else {
DebugPrintf("Syntax: cmd_showframe <seqname> <index> (<cd number>)\n");
}
return true;
}
/**
* Command: shows a background
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdShowBg(int argc, const char **argv) {
if (argc == 2 || argc == 3) {
Common::String filename(const_cast<char *>(argv[1]));
if (argc == 3)
loadArchive((ArchiveIndex)getNumber(argv[2]));
if (!_engine->getResourceManager()->hasFile(filename + ".BG")) {
DebugPrintf("Cannot find file: %s\n", (filename + ".BG").c_str());
return true;
}
// Store command
if (!hasCommand()) {
_command = WRAP_METHOD(Debugger, cmdShowBg);
copyCommand(argc, argv);
return Cmd_Exit(0, 0);
} else {
clearBg(GraphicsManager::kBackgroundC);
Background *background = _engine->getResourceManager()->loadBackground(filename);
if (background) {
_engine->getGraphicsManager()->draw(background, GraphicsManager::kBackgroundC);
delete background;
askForRedraw();
}
redrawScreen();
if (argc == 3)
restoreArchive();
// Pause for a second to be able to see the background
_engine->_system->delayMillis(1000);
resetCommand();
}
} else {
DebugPrintf("Syntax: showbg <bgname> (<cd number>)\n");
}
return true;
}
/**
* Command: plays a sequence.
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdPlaySeq(int argc, const char **argv) {
if (argc == 2 || argc == 3) {
Common::String filename(const_cast<char *>(argv[1]));
filename += ".seq";
if (argc == 3)
loadArchive((ArchiveIndex)getNumber(argv[2]));
if (!_engine->getResourceManager()->hasFile(filename)) {
DebugPrintf("Cannot find file: %s\n", filename.c_str());
return true;
}
// Store command
if (!hasCommand()) {
_command = WRAP_METHOD(Debugger, cmdPlaySeq);
copyCommand(argc, argv);
return Cmd_Exit(0, 0);
} else {
Sequence *sequence = new Sequence(filename);
if (sequence->load(getArchive(filename))) {
// Check that we have at least a frame to show
if (sequence->count() == 0) {
delete sequence;
return false;
}
_engine->getCursor()->show(false);
SequenceFrame player(sequence, 0, true);
do {
// Clear screen
clearBg(GraphicsManager::kBackgroundA);
_engine->getGraphicsManager()->draw(&player, GraphicsManager::kBackgroundA);
askForRedraw();
redrawScreen();
// Handle right-click to interrupt sequence
Common::Event ev;
_engine->getEventManager()->pollEvent(ev);
if (ev.type == Common::EVENT_RBUTTONUP)
break;
_engine->_system->delayMillis(175);
// go to the next frame
} while (player.nextFrame());
_engine->getCursor()->show(true);
} else {
// Sequence player is deleting his reference to the sequence, but we need to take care of it if the
// sequence could not be loaded
delete sequence;
}
resetCommand();
if (argc == 3)
restoreArchive();
}
} else {
DebugPrintf("Syntax: playseq <seqname> (<cd number>)\n");
}
return true;
}
/**
* Command: plays a sound
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdPlaySnd(int argc, const char **argv) {
if (argc == 2 || argc == 3) {
if (argc == 3)
loadArchive((ArchiveIndex)getNumber(argv[2]));
// Add .SND at the end of the filename if needed
Common::String name(const_cast<char *>(argv[1]));
if (!name.contains('.'))
name += ".SND";
if (!_engine->getResourceManager()->hasFile(name)) {
DebugPrintf("Cannot find file: %s\n", name.c_str());
return true;
}
_engine->_system->getMixer()->stopAll();
_soundStream->load(getArchive(name));
if (argc == 3)
restoreArchive();
} else {
DebugPrintf("Syntax: playsnd <sndname> (<cd number>)\n");
}
return true;
}
/**
* Command: plays subtitles
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdPlaySbe(int argc, const char **argv) {
if (argc == 2 || argc == 3) {
Common::String filename(const_cast<char *>(argv[1]));
if (argc == 3)
loadArchive((ArchiveIndex)getNumber(argv[2]));
filename += ".sbe";
if (!_engine->getResourceManager()->hasFile(filename)) {
DebugPrintf("Cannot find file: %s\n", filename.c_str());
return true;
}
// Store command
if (!hasCommand()) {
_command = WRAP_METHOD(Debugger, cmdPlaySbe);
copyCommand(argc, argv);
return Cmd_Exit(0, 0);
} else {
SubtitleManager subtitle(_engine->getFont());
if (subtitle.load(getArchive(filename))) {
_engine->getCursor()->show(false);
for (uint16 i = 0; i < subtitle.getMaxTime(); i += 25) {
clearBg(GraphicsManager::kBackgroundAll);
subtitle.setTime(i);
_engine->getGraphicsManager()->draw(&subtitle, GraphicsManager::kBackgroundOverlay);
askForRedraw();
redrawScreen();
// Handle right-click to interrupt sequence
Common::Event ev;
_engine->getEventManager()->pollEvent(ev);
if (ev.type == Common::EVENT_RBUTTONUP)
break;
_engine->_system->delayMillis(500);
}
_engine->getCursor()->show(true);
}
if (argc == 3)
restoreArchive();
resetCommand();
}
} else {
DebugPrintf("Syntax: playsbe <sbename> (<cd number>)\n");
}
return true;
}
/**
* Command: plays a NIS animation sequence.
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdPlayNis(int argc, const char **argv) {
if (argc == 2 || argc == 3) {
Common::String name(const_cast<char *>(argv[1]));
if (argc == 3)
loadArchive((ArchiveIndex)getNumber(argv[2]));
// If we got a nis filename, check that the file exists
if (name.contains('.') && _engine->getResourceManager()->hasFile(name)) {
DebugPrintf("Cannot find file: %s\n", name.c_str());
return true;
}
// Store command
if (!hasCommand()) {
_command = WRAP_METHOD(Debugger, cmdPlayNis);
copyCommand(argc, argv);
return Cmd_Exit(0, 0);
} else {
// Make sure we are not called in a loop
_numParams = 0;
// Check if we got a nis filename or an animation index
if (name.contains('.')) {
Animation animation;
if (animation.load(getArchive(name))) {
_engine->getCursor()->show(false);
animation.play();
_engine->getCursor()->show(true);
}
} else {
getAction()->playAnimation((EventIndex)atoi(name.c_str()), true);
}
if (argc == 3)
restoreArchive();
resetCommand();
}
} else {
DebugPrintf("Syntax: playnis <nisname.nis or animation index> (<cd number>)\n");
}
return true;
}
/**
* Command: loads a scene
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdLoadScene(int argc, const char **argv) {
if (argc == 2 || argc == 3) {
int cd = 1;
SceneIndex index = (SceneIndex)getNumber(argv[1]);
// Check args
if (argc == 3)
loadArchive((ArchiveIndex)getNumber(argv[2]));
if (index > 2500) {
DebugPrintf("Error: invalid index value (0-2500)");
return true;
}
// Store command
if (!hasCommand()) {
_command = WRAP_METHOD(Debugger, cmdLoadScene);
copyCommand(argc, argv);
return Cmd_Exit(0, 0);
} else {
clearBg(GraphicsManager::kBackgroundAll);
/************ DEBUG *************************/
// Use to find scenes with certain values
//for (int i = index; i < 2500; i++) {
// loadSceneObject(scene, i);
// if (scene.getHeader() && scene.getHeader()->car == 5 && scene.getHeader()->position == 81) {
// DebugPrintf("Found scene: %d", i);
// // Draw scene found
// _engine->getGraphicsManager()->draw(&scene, GraphicsManager::kBackgroundC);
// askForRedraw();
// redrawScreen();
// _engine->_system->delayMillis(500);
// break;
// }
//}
//delete _sceneLoader;
//resetCommand();
//return true;
/*********************************************/
Scene *scene = getScenes()->get(index);
if (!scene) {
DebugPrintf("Cannot load scene %i from CD %i", index, cd);
resetCommand();
return true;
}
_engine->getGraphicsManager()->draw(scene, GraphicsManager::kBackgroundC);
askForRedraw();
redrawScreen();
// Pause for a second to be able to see the scene
_engine->_system->delayMillis(500);
if (argc == 3)
restoreArchive();
resetCommand();
}
} else {
DebugPrintf("Syntax: loadscene <scene index> (<cd number>)\n");
}
return true;
}
/**
* Command: starts a fight sequence
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdFight(int argc, const char **argv) {
if (argc == 2) {
FightType type = (FightType)getNumber(argv[1]);
// Load proper data file
ArchiveIndex index = kArchiveCd1;
switch (type) {
default:
goto error;
case kFightMilos:
index = kArchiveCd1;
break;
case kFightAnna:
index = kArchiveCd2;
break;
case kFightIvo:
case kFightSalko:
case kFightVesna:
index = kArchiveCd3;
break;
}
loadArchive(index);
// Store command
if (!hasCommand()) {
_command = WRAP_METHOD(Debugger, cmdFight);
copyCommand(argc, argv);
return false;
} else {
// Make sure we are not called in a loop
_numParams = 0;
clearBg(GraphicsManager::kBackgroundAll);
askForRedraw();
redrawScreen();
SceneIndex lastScene = getState()->scene;
getFight()->setup(type) ? DebugPrintf("Lost fight!\n") : DebugPrintf("Won fight!\n");
// Pause for a second to be able to see the final scene
_engine->_system->delayMillis(1000);
// Restore loaded archive
restoreArchive();
// Stop audio and restore scene
getSoundQueue()->stopAllSound();
clearBg(GraphicsManager::kBackgroundAll);
Scene *scene = getScenes()->get(lastScene);
_engine->getGraphicsManager()->draw(scene, GraphicsManager::kBackgroundC);
askForRedraw();
redrawScreen();
resetCommand();
}
} else {
error:
DebugPrintf("Syntax: fight <id> (id=2001-2005)\n");
}
return true;
}
/**
* Command: starts the beetle sequence
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdBeetle(int argc, const char **argv) {
if (argc == 1) {
// Load proper data file (beetle game in in Cd2)
loadArchive(kArchiveCd2);
// Store command
if (!hasCommand()) {
_command = WRAP_METHOD(Debugger, cmdBeetle);
copyCommand(argc, argv);
return false;
} else {
clearBg(GraphicsManager::kBackgroundAll);
askForRedraw();
redrawScreen();
// Save current state
SceneIndex previousScene = getState()->scene;
ObjectLocation previousLocation = getInventory()->get(kItemBeetle)->location;
ChapterIndex previousChapter = (ChapterIndex)getProgress().chapter;
// Setup scene & inventory
getProgress().chapter = kChapter2;
Scene *scene = getScenes()->get(kSceneBeetle);
getInventory()->get(kItemBeetle)->location = kObjectLocation3;
askForRedraw();
redrawScreen();
// Load the beetle game
Action *action = NULL;
Beetle *beetle = new Beetle(_engine);
if (!beetle->isLoaded())
beetle->load();
// Play the game
Common::Event ev;
bool playgame = true;
while (playgame) {
// Update beetle
beetle->update();
askForRedraw();
redrawScreen();
while (g_system->getEventManager()->pollEvent(ev)) {
switch (ev.type) {
default:
break;
case Common::EVENT_KEYDOWN:
// Exit beetle game on escape
if (ev.kbd.keycode == Common::KEYCODE_ESCAPE)
playgame = false;
break;
case Common::EVENT_MOUSEMOVE: {
// Update cursor
CursorStyle style = kCursorNormal;
SceneHotspot *hotspot = NULL;
if (scene->checkHotSpot(ev.mouse, &hotspot)) {
if (!action)
action = new Action(_engine);
style = action->getCursor(*hotspot);
}
_engine->getCursor()->setStyle(style);
break;
}
case Common::EVENT_LBUTTONUP:
case Common::EVENT_RBUTTONUP:
// Update coordinates
getLogic()->getGameState()->setCoordinates(ev.mouse);
if (beetle->catchBeetle())
playgame = false;
break;
}
_engine->_system->delayMillis(10);
}
}
// Cleanup
beetle->unload();
delete beetle;
delete action;
// Pause for a second to be able to see the final scene
_engine->_system->delayMillis(1000);
// Restore state
getProgress().chapter = previousChapter;
getInventory()->get(kItemBeetle)->location = previousLocation;
// Restore loaded archive
restoreArchive();
// Stop audio and restore scene
getSoundQueue()->stopAllSound();
clearBg(GraphicsManager::kBackgroundAll);
Scene *oldScene = getScenes()->get(previousScene);
_engine->getGraphicsManager()->draw(oldScene, GraphicsManager::kBackgroundC);
askForRedraw();
redrawScreen();
resetCommand();
}
} else {
DebugPrintf("Syntax: beetle\n");
}
return true;
}
/**
* Command: adjusts the time delta
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdTimeDelta(int argc, const char **argv) {
if (argc == 2) {
int delta = getNumber(argv[1]);
if (delta <= 0 || delta > 500)
goto label_error;
getState()->timeDelta = (uint)delta;
} else {
label_error:
DebugPrintf("Syntax: delta <time delta> (delta=1-500)\n");
}
return true;
}
/**
* Command: Convert between in-game time and human readable time
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdTime(int argc, const char **argv) {
if (argc == 2) {
int32 time = getNumber(argv[1]);
if (time < 0)
goto label_error;
// Convert to human-readable form
uint8 hours = 0;
uint8 minutes = 0;
State::getHourMinutes((uint32)time, &hours, &minutes);
DebugPrintf("%02d:%02d\n", hours, minutes);
} else {
label_error:
DebugPrintf("Syntax: time <time to convert> (time=0-INT_MAX)\n");
}
return true;
}
/**
* Command: show game logic data
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdShow(int argc, const char **argv) {
#define OUTPUT_DUMP(name, text) \
DebugPrintf(#name "\n"); \
DebugPrintf("--------------------------------------------------------------------\n\n"); \
DebugPrintf("%s", text); \
DebugPrintf("\n");
if (argc == 2) {
Common::String name(const_cast<char *>(argv[1]));
if (name == "state" || name == "st") {
OUTPUT_DUMP("Game state", getState()->toString().c_str());
} else if (name == "progress" || name == "pr") {
OUTPUT_DUMP("Progress", getProgress().toString().c_str());
} else if (name == "flags" || name == "fl") {
OUTPUT_DUMP("Flags", getFlags()->toString().c_str());
} else if (name == "inventory" || name == "inv") {
OUTPUT_DUMP("Inventory", getInventory()->toString().c_str());
} else if (name == "objects" || name == "obj") {
OUTPUT_DUMP("Objects", getObjects()->toString().c_str());
} else if (name == "savepoints" || name == "pt") {
OUTPUT_DUMP("SavePoints", getSavePoints()->toString().c_str());
} else if (name == "scene" || name == "sc") {
OUTPUT_DUMP("Current scene", getScenes()->get(getState()->scene)->toString().c_str());
} else {
goto label_error;
}
} else {
label_error:
DebugPrintf("Syntax: state <option>\n");
DebugPrintf(" state / st\n");
DebugPrintf(" progress / pr\n");
DebugPrintf(" flags / fl\n");
DebugPrintf(" inventory / inv\n");
DebugPrintf(" objects / obj\n");
DebugPrintf(" savepoints / pt\n");
DebugPrintf(" scene / sc\n");
}
return true;
#undef OUTPUT_DUMP
}
/**
* Command: shows entity data
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdEntity(int argc, const char **argv) {
if (argc == 2) {
EntityIndex index = (EntityIndex)getNumber(argv[1]);
if (index > 39)
goto label_error;
DebugPrintf("Entity %s\n", ENTITY_NAME(index));
DebugPrintf("--------------------------------------------------------------------\n\n");
DebugPrintf("%s", getEntities()->getData(index)->toString().c_str());
// The Player entity does not have any callback data
if (index != kEntityPlayer) {
EntityData *data = getEntities()->get(index)->getParamData();
for (uint callback = 0; callback < 9; callback++) {
DebugPrintf("Call parameters %d:\n", callback);
for (byte ix = 0; ix < 4; ix++)
DebugPrintf(" %s", data->getParameters(callback, ix)->toString().c_str());
}
}
DebugPrintf("\n");
} else {
label_error:
DebugPrintf("Syntax: entity <index>\n");
for (int i = 0; i < 40; i += 4)
DebugPrintf(" %s - %d %s - %d %s - %d %s - %d\n", ENTITY_NAME(i), i, ENTITY_NAME(i+1), i+1, ENTITY_NAME(i+2), i+2, ENTITY_NAME(i+3), i+3);
}
return true;
}
/**
* Command: loads a game
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdLoadGame(int argc, const char **argv) {
if (argc == 2) {
int id = getNumber(argv[1]);
if (id == 0 || id > 6)
goto error;
getSaveLoad()->loadGame((GameId)(id - 1));
} else {
error:
DebugPrintf("Syntax: loadgame <id> (id=1-6)\n");
}
return true;
}
/**
* Command: switch to a specific chapter
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdSwitchChapter(int argc, const char **argv) {
if (argc == 2) {
int id = getNumber(argv[1]);
if (id <= 1 || id > 6)
goto error;
// Store command
if (!hasCommand()) {
_command = WRAP_METHOD(Debugger, cmdSwitchChapter);
copyCommand(argc, argv);
return false;
} else {
// Sets the current chapter and then call Logic::switchChapter to proceed to the next chapter
getState()->progress.chapter = (ChapterIndex)(id - 1);
getLogic()->switchChapter();
resetCommand();
}
} else {
error:
DebugPrintf("Syntax: chapter <id> (id=2-6)\n");
}
return true;
}
/**
* Command: clears the screen
*
* @param argc The argument count.
* @param argv The values.
*
* @return true if it was handled, false otherwise
*/
bool Debugger::cmdClear(int argc, const char **) {
if (argc == 1) {
clearBg(GraphicsManager::kBackgroundAll);
askForRedraw();
redrawScreen();
} else {
DebugPrintf("Syntax: clear - clear the screen\n");
}
return true;
}
} // End of namespace LastExpress