aboutsummaryrefslogtreecommitdiff
path: root/engines/agi/saveload.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'engines/agi/saveload.cpp')
-rw-r--r--engines/agi/saveload.cpp1104
1 files changed, 551 insertions, 553 deletions
diff --git a/engines/agi/saveload.cpp b/engines/agi/saveload.cpp
index 41a7a943ff..5f1c5112cd 100644
--- a/engines/agi/saveload.cpp
+++ b/engines/agi/saveload.cpp
@@ -38,33 +38,44 @@
#include "agi/agi.h"
#include "agi/graphics.h"
+#include "agi/text.h"
#include "agi/sprite.h"
#include "agi/keyboard.h"
#include "agi/menu.h"
+#include "agi/systemui.h"
+#include "agi/words.h"
-#define SAVEGAME_VERSION 6
+#define SAVEGAME_CURRENT_VERSION 10
//
-// Version 0 (Sarien): view table has 64 entries
-// Version 1 (Sarien): view table has 256 entries (needed in KQ3)
-// Version 2 (ScummVM): first ScummVM version
-// Version 3 (ScummVM): added AGIPAL save/load support
-// Version 4 (ScummVM): added thumbnails and save creation date/time
-// Version 5 (ScummVM): Added game md5
-// Version 6 (ScummVM): Added game played time
-//
+// Version 0 (Sarien): view table has 64 entries
+// Version 1 (Sarien): view table has 256 entries (needed in KQ3)
+// Version 2 (ScummVM): first ScummVM version
+// Version 3 (ScummVM): added AGIPAL save/load support
+// Version 4 (ScummVM): added thumbnails and save creation date/time
+// Version 5 (ScummVM): Added game md5
+// Version 6 (ScummVM): Added game played time
+// Version 7 (ScummVM): Added controller key mappings
+// required for some games for quick-loading from ScummVM main menu
+// for games, that do not set all key mappings right at the start
+// Added automatic save data (for command SetSimple)
+// Version 8 (ScummVM): Added Hold-Key-Mode boolean
+// required for at least Mixed Up Mother Goose
+// gets set at the start of the game only
+// Version 9 (ScummVM): Added seconds to saved game time stamp
+// Version 10 (ScummVM): Added priorityTableSet boolean
namespace Agi {
-static const uint32 AGIflag = MKTAG('A','G','I',':');
+static const uint32 AGIflag = MKTAG('A', 'G', 'I', ':');
-int AgiEngine::saveGame(const Common::String &fileName, const Common::String &description) {
+int AgiEngine::saveGame(const Common::String &fileName, const Common::String &descriptionString) {
char gameIDstring[8] = "gameIDX";
int i;
Common::OutSaveFile *out;
int result = errOK;
- debugC(3, kDebugLevelMain | kDebugLevelSavegame, "AgiEngine::saveGame(%s, %s)", fileName.c_str(), description.c_str());
+ debugC(3, kDebugLevelMain | kDebugLevelSavegame, "AgiEngine::saveGame(%s, %s)", fileName.c_str(), descriptionString.c_str());
if (!(out = _saveFileMan->openForSaving(fileName))) {
warning("Can't create file '%s', game not saved", fileName.c_str());
return errBadFileOpen;
@@ -73,10 +84,17 @@ int AgiEngine::saveGame(const Common::String &fileName, const Common::String &de
}
out->writeUint32BE(AGIflag);
- out->write(description.c_str(), 31);
- out->writeByte(SAVEGAME_VERSION);
- debugC(5, kDebugLevelMain | kDebugLevelSavegame, "Writing save game version (%d)", SAVEGAME_VERSION);
+ // Write description of saved game, limited to SAVEDGAME_DESCRIPTION_LEN characters + terminating NUL
+ char description[SAVEDGAME_DESCRIPTION_LEN + 1];
+
+ memset(description, 0, sizeof(description));
+ strncpy(description, descriptionString.c_str(), SAVEDGAME_DESCRIPTION_LEN);
+ assert(SAVEDGAME_DESCRIPTION_LEN + 1 == 31); // safety
+ out->write(description, 31);
+
+ out->writeByte(SAVEGAME_CURRENT_VERSION);
+ debugC(5, kDebugLevelMain | kDebugLevelSavegame, "Writing save game version (%d)", SAVEGAME_CURRENT_VERSION);
// Thumbnail
Graphics::saveThumbnail(*out);
@@ -93,11 +111,12 @@ int AgiEngine::saveGame(const Common::String &fileName, const Common::String &de
debugC(5, kDebugLevelMain | kDebugLevelSavegame, "Writing save date (%d)", saveDate);
out->writeUint16BE(saveTime);
debugC(5, kDebugLevelMain | kDebugLevelSavegame, "Writing save time (%d)", saveTime);
+ // Version 9+: save seconds of current time as well
+ out->writeByte(curTime.tm_sec & 0xFF);
out->writeUint32BE(playTime);
debugC(5, kDebugLevelMain | kDebugLevelSavegame, "Writing play time (%d)", playTime);
- out->writeByte(_game.state);
- debugC(5, kDebugLevelMain | kDebugLevelSavegame, "Writing game state (%d)", _game.state);
+ out->writeByte(2); // was _game.state, 2 = STATE_RUNNING
strcpy(gameIDstring, _game.id);
out->write(gameIDstring, 8);
@@ -116,37 +135,56 @@ int AgiEngine::saveGame(const Common::String &fileName, const Common::String &de
out->writeByte(tmp[i]);
}
+ // Version 7+: Save automatic saving state (set.simple opcode)
+ out->writeByte(_game.automaticSave);
+ out->write(_game.automaticSaveDescription, 31);
+
+ // touch VM_VAR_SECONDS, so that it gets updated
+ getVar(VM_VAR_SECONDS);
+
for (i = 0; i < MAX_FLAGS; i++)
out->writeByte(_game.flags[i]);
for (i = 0; i < MAX_VARS; i++)
out->writeByte(_game.vars[i]);
out->writeSint16BE((int8)_game.horizon);
- out->writeSint16BE((int16)_game.lineStatus);
- out->writeSint16BE((int16)_game.lineUserInput);
- out->writeSint16BE((int16)_game.lineMinPrint);
+ out->writeSint16BE((int16)_text->statusRow_Get());
+ out->writeSint16BE((int16)_text->promptRow_Get());
+ out->writeSint16BE((int16)_text->getWindowRowMin());
- out->writeSint16BE((int16)_game.inputMode);
- out->writeSint16BE((int16)_game.lognum);
+ out->writeSint16BE(1); // was _game.inputMode, we set it to 1, which was INPUTMODE_NORMAL
+ out->writeSint16BE((int16)_game.curLogicNr);
out->writeSint16BE((int16)_game.playerControl);
out->writeSint16BE((int16)shouldQuit());
- out->writeSint16BE((int16)_game.statusLine);
- out->writeSint16BE((int16)_game.clockEnabled);
+ if (_text->statusEnabled()) {
+ out->writeSint16BE(0x7FFF);
+ } else {
+ out->writeSint16BE(0);
+ }
+ out->writeSint16BE(1); // was clock enabled
+ // (previous in-game-timer, in-game-timer is always enabled during the regular game, so need to save/load it)
out->writeSint16BE((int16)_game.exitAllLogics);
out->writeSint16BE((int16)_game.pictureShown);
- out->writeSint16BE((int16)_game.hasPrompt);
+ out->writeSint16BE((int16)_text->promptIsEnabled()); // was "_game.hasPrompt", no longer needed
out->writeSint16BE((int16)_game.gameFlags);
- out->writeSint16BE(_game.inputEnabled);
+ if (_text->promptIsEnabled()) {
+ out->writeSint16BE(0x7FFF);
+ } else {
+ out->writeSint16BE(0);
+ }
+
+ for (i = 0; i < SCRIPT_HEIGHT; i++)
+ out->writeByte(_gfx->saveLoadGetPriority(i));
- for (i = 0; i < _HEIGHT; i++)
- out->writeByte(_game.priTable[i]);
+ // Version 10+: Save, if priority table got modified (set.pri.base opcode)
+ out->writeSint16BE((int16)_gfx->saveLoadWasPriorityTableModified());
out->writeSint16BE((int16)_game.gfxMode);
- out->writeByte(_game.cursorChar);
- out->writeSint16BE((int16)_game.colorFg);
- out->writeSint16BE((int16)_game.colorBg);
+ out->writeByte(_text->inputGetCursorChar());
+ out->writeSint16BE((int16)_text->charAttrib_GetForeground());
+ out->writeSint16BE((int16)_text->charAttrib_GetBackground());
// game.hires
// game.sbuf
@@ -157,21 +195,33 @@ int AgiEngine::saveGame(const Common::String &fileName, const Common::String &de
for (i = 0; i < (int16)_game.numObjects; i++)
out->writeSint16BE((int16)objectGetLocation(i));
+ // Version 7+: save controller key mappings
+ // required for games, that do not set all key mappings right at the start
+ // when quick restoring is used from ScummVM menu, only 1 cycle is executed
+ for (i = 0; i < MAX_CONTROLLER_KEYMAPPINGS; i++) {
+ out->writeUint16BE(_game.controllerKeyMapping[i].keycode);
+ out->writeByte(_game.controllerKeyMapping[i].controllerSlot);
+ }
+
+ // Version 8+: hold-key-mode
+ // required for at least Mixed Up Mother Goose
+ out->writeByte(_keyHoldMode);
+
// game.ev_keyp
for (i = 0; i < MAX_STRINGS; i++)
out->write(_game.strings[i], MAX_STRINGLEN);
// record info about loaded resources
- for (i = 0; i < MAX_DIRS; i++) {
+ for (i = 0; i < MAX_DIRECTORY_ENTRIES; i++) {
out->writeByte(_game.dirLogic[i].flags);
out->writeSint16BE((int16)_game.logics[i].sIP);
out->writeSint16BE((int16)_game.logics[i].cIP);
}
- for (i = 0; i < MAX_DIRS; i++)
+ for (i = 0; i < MAX_DIRECTORY_ENTRIES; i++)
out->writeByte(_game.dirPic[i].flags);
- for (i = 0; i < MAX_DIRS; i++)
+ for (i = 0; i < MAX_DIRECTORY_ENTRIES; i++)
out->writeByte(_game.dirView[i].flags);
- for (i = 0; i < MAX_DIRS; i++)
+ for (i = 0; i < MAX_DIRECTORY_ENTRIES; i++)
out->writeByte(_game.dirSound[i].flags);
// game.pictures
@@ -179,51 +229,77 @@ int AgiEngine::saveGame(const Common::String &fileName, const Common::String &de
// game.views
// game.sounds
- for (i = 0; i < MAX_VIEWTABLE; i++) {
- VtEntry *v = &_game.viewTable[i];
+ for (i = 0; i < SCREENOBJECTS_MAX; i++) {
+ ScreenObjEntry *screenObj = &_game.screenObjTable[i];
- out->writeByte(v->stepTime);
- out->writeByte(v->stepTimeCount);
- out->writeByte(v->entry);
- out->writeSint16BE(v->xPos);
- out->writeSint16BE(v->yPos);
- out->writeByte(v->currentView);
+ out->writeByte(screenObj->stepTime);
+ out->writeByte(screenObj->stepTimeCount);
+ out->writeByte(screenObj->objectNr);
+ out->writeSint16BE(screenObj->xPos);
+ out->writeSint16BE(screenObj->yPos);
+ out->writeByte(screenObj->currentViewNr);
// v->view_data
- out->writeByte(v->currentLoop);
- out->writeByte(v->numLoops);
+ out->writeByte(screenObj->currentLoopNr);
+ out->writeByte(screenObj->loopCount);
// v->loop_data
- out->writeByte(v->currentCel);
- out->writeByte(v->numCels);
+ out->writeByte(screenObj->currentCelNr);
+ out->writeByte(screenObj->celCount);
// v->cel_data
// v->cel_data_2
- out->writeSint16BE(v->xPos2);
- out->writeSint16BE(v->yPos2);
+ out->writeSint16BE(screenObj->xPos_prev);
+ out->writeSint16BE(screenObj->yPos_prev);
// v->s
- out->writeSint16BE(v->xSize);
- out->writeSint16BE(v->ySize);
- out->writeByte(v->stepSize);
- out->writeByte(v->cycleTime);
- out->writeByte(v->cycleTimeCount);
- out->writeByte(v->direction);
+ out->writeSint16BE(screenObj->xSize);
+ out->writeSint16BE(screenObj->ySize);
+ out->writeByte(screenObj->stepSize);
+ out->writeByte(screenObj->cycleTime);
+ out->writeByte(screenObj->cycleTimeCount);
+ out->writeByte(screenObj->direction);
- out->writeByte(v->motion);
- out->writeByte(v->cycle);
- out->writeByte(v->priority);
+ out->writeByte(screenObj->motionType);
+ out->writeByte(screenObj->cycle);
+ out->writeByte(screenObj->priority);
- out->writeUint16BE(v->flags);
+ out->writeUint16BE(screenObj->flags);
- out->writeByte(v->parm1);
- out->writeByte(v->parm2);
- out->writeByte(v->parm3);
- out->writeByte(v->parm4);
+ // this was done so that saved games compatibility isn't broken
+ switch (screenObj->motionType) {
+ case kMotionNormal:
+ out->writeByte(0);
+ out->writeByte(0);
+ out->writeByte(0);
+ out->writeByte(0);
+ break;
+ case kMotionWander:
+ out->writeByte(screenObj->wander_count);
+ out->writeByte(0);
+ out->writeByte(0);
+ out->writeByte(0);
+ break;
+ case kMotionFollowEgo:
+ out->writeByte(screenObj->follow_stepSize);
+ out->writeByte(screenObj->follow_flag);
+ out->writeByte(screenObj->follow_count);
+ out->writeByte(0);
+ break;
+ case kMotionEgo:
+ case kMotionMoveObj:
+ out->writeByte((byte)screenObj->move_x); // problematic! int16 -> byte
+ out->writeByte((byte)screenObj->move_y);
+ out->writeByte(screenObj->move_stepSize);
+ out->writeByte(screenObj->move_flag);
+ break;
+ default:
+ error("unknown motion-type");
+ }
}
// Save image stack
@@ -249,7 +325,7 @@ int AgiEngine::saveGame(const Common::String &fileName, const Common::String &de
warning("Can't write file '%s'. (Disk full?)", fileName.c_str());
result = errIOError;
} else
- debugC(1, kDebugLevelMain | kDebugLevelSavegame, "Saved game %s in file %s", description.c_str(), fileName.c_str());
+ debugC(1, kDebugLevelMain | kDebugLevelSavegame, "Saved game %s in file %s", descriptionString.c_str(), fileName.c_str());
delete out;
debugC(3, kDebugLevelMain | kDebugLevelSavegame, "Closed %s", fileName.c_str());
@@ -260,11 +336,14 @@ int AgiEngine::saveGame(const Common::String &fileName, const Common::String &de
}
int AgiEngine::loadGame(const Common::String &fileName, bool checkId) {
- char description[31], saveVersion, loadId[8];
- int i, vtEntries = MAX_VIEWTABLE;
+ char description[SAVEDGAME_DESCRIPTION_LEN + 1];
+ byte saveVersion = 0;
+ char loadId[8];
+ int i, vtEntries = SCREENOBJECTS_MAX;
uint8 t;
int16 parm[7];
Common::InSaveFile *in;
+ bool totalPlayTimeWasSet = false;
debugC(3, kDebugLevelMain | kDebugLevelSavegame, "AgiEngine::loadGame(%s)", fileName.c_str());
@@ -284,30 +363,45 @@ int AgiEngine::loadGame(const Common::String &fileName, bool checkId) {
return errOK;
}
- in->read(description, 31);
+ assert(SAVEDGAME_DESCRIPTION_LEN + 1 == 31); // safety
+ in->read(description, 31); // skip description
+ // check, if there is a terminating NUL inside description
+ uint16 descriptionPos = 0;
+ while (description[descriptionPos]) {
+ descriptionPos++;
+ if (descriptionPos >= sizeof(description))
+ error("saved game description is corrupt");
+ }
debugC(6, kDebugLevelMain | kDebugLevelSavegame, "Description is: %s", description);
saveVersion = in->readByte();
- if (saveVersion < 2) // is the save game pre-ScummVM?
- warning("Old save game version (%d, current version is %d). Will try and read anyway, but don't be surprised if bad things happen", saveVersion, SAVEGAME_VERSION);
+ if (saveVersion < 2) // is the save game pre-ScummVM?
+ warning("Old save game version (%d, current version is %d). Will try and read anyway, but don't be surprised if bad things happen", saveVersion, SAVEGAME_CURRENT_VERSION);
if (saveVersion < 3)
warning("This save game contains no AGIPAL data, if the game is using the AGIPAL hack, it won't work correctly");
+ if (saveVersion > SAVEGAME_CURRENT_VERSION)
+ error("Saved game was created with a newer version of ScummVM. Unable to load.");
+
if (saveVersion >= 4) {
// We don't need the thumbnail here, so just read it and discard it
Graphics::skipThumbnail(*in);
- in->readUint32BE(); // save date
- in->readUint16BE(); // save time
+ in->readUint32BE(); // save date
+ in->readUint16BE(); // save time (hour + minute)
+ if (saveVersion >= 9) {
+ in->readByte(); // save time seconds
+ }
if (saveVersion >= 6) {
uint32 playTime = in->readUint32BE();
- g_engine->setTotalPlayTime(playTime * 1000);
+ inGameTimerReset(playTime * 1000);
+ totalPlayTimeWasSet = true;
}
}
- _game.state = (State)in->readByte();
+ in->readByte(); // was _game.state, not needed anymore
in->read(loadId, 8);
if (strcmp(loadId, _game.id) != 0 && checkId) {
@@ -348,56 +442,93 @@ int AgiEngine::loadGame(const Common::String &fileName, bool checkId) {
}
}
+ if (saveVersion >= 7) {
+ // Restore automatic saving state (set.simple opcode)
+ _game.automaticSave = in->readByte();
+ in->read(_game.automaticSaveDescription, 31);
+ } else {
+ _game.automaticSave = false;
+ _game.automaticSaveDescription[0] = 0;
+ }
+
for (i = 0; i < MAX_FLAGS; i++)
_game.flags[i] = in->readByte();
for (i = 0; i < MAX_VARS; i++)
_game.vars[i] = in->readByte();
- setvar(vFreePages, 180); // Set amount of free memory to realistic value (Overwriting the just loaded value)
+ if (!totalPlayTimeWasSet) {
+ // If we haven't gotten total play time by now, try to calculate it by using VM Variables
+ // This will happen for at least saves before version 6
+ // Direct access because otherwise we would trigger an update to these variables according to ScummVM total play time
+ byte playTimeSeconds = _game.vars[VM_VAR_SECONDS];
+ byte playTimeMinutes = _game.vars[VM_VAR_MINUTES];
+ byte playTimeHours = _game.vars[VM_VAR_HOURS];
+ byte playTimeDays = _game.vars[VM_VAR_DAYS];
+ uint32 playTime = (playTimeSeconds + (playTimeMinutes * 60) + (playTimeHours * 3600) + (playTimeDays * 86400)) * 1000;
+
+ inGameTimerReset(playTime);
+ }
+
+ setVar(VM_VAR_FREE_PAGES, 180); // Set amount of free memory to realistic value (Overwriting the just loaded value)
_game.horizon = in->readSint16BE();
- _game.lineStatus = in->readSint16BE();
- _game.lineUserInput = in->readSint16BE();
- _game.lineMinPrint = in->readSint16BE();
+ _text->statusRow_Set(in->readSint16BE());
+ _text->promptRow_Set(in->readSint16BE());
+ _text->configureScreen(in->readSint16BE());
// These are never saved
- _game.cursorPos = 0;
- _game.inputBuffer[0] = 0;
- _game.echoBuffer[0] = 0;
- _game.keypress = 0;
+ _text->promptReset();
+
+ in->readSint16BE(); // was _game.inputMode, not needed anymore
- _game.inputMode = (InputMode)in->readSint16BE();
- _game.lognum = in->readSint16BE();
+ _game.curLogicNr = in->readSint16BE();
_game.playerControl = in->readSint16BE();
if (in->readSint16BE())
quitGame();
- _game.statusLine = in->readSint16BE();
- _game.clockEnabled = in->readSint16BE();
+ if (in->readSint16BE()) {
+ _text->statusEnable();
+ } else {
+ _text->statusDisable();
+ }
+ in->readSint16BE(); // was clock enabled, no longer needed
_game.exitAllLogics = in->readSint16BE();
- _game.pictureShown = in->readSint16BE();
- _game.hasPrompt = in->readSint16BE();
+ in->readSint16BE(); // was _game.pictureShown
+ in->readSint16BE(); // was _game.hasPrompt, no longer needed
_game.gameFlags = in->readSint16BE();
- _game.inputEnabled = in->readSint16BE();
+ if (in->readSint16BE()) {
+ _text->promptEnable();
+ } else {
+ _text->promptDisable();
+ }
+
+ for (i = 0; i < SCRIPT_HEIGHT; i++)
+ _gfx->saveLoadSetPriority(i, in->readByte());
+
+ if (saveVersion >= 10) {
+ // Version 10+: priority table was modified by scripts
+ int16 priorityTableWasModified = in->readSint16BE();
- for (i = 0; i < _HEIGHT; i++)
- _game.priTable[i] = in->readByte();
+ if (priorityTableWasModified) {
+ _gfx->saveLoadSetPriorityTableModifiedBool(true);
+ } else {
+ _gfx->saveLoadSetPriorityTableModifiedBool(false);
+ }
+ } else {
+ // Try to figure it out by ourselves
+ _gfx->saveLoadFigureOutPriorityTableModifiedBool();
+ }
- if (_game.hasWindow)
- closeWindow();
+ _text->closeWindow();
- _game.msgBoxTicks = 0;
_game.block.active = false;
- // game.window - fixed by close_window()
- // game.has_window - fixed by close_window()
_game.gfxMode = in->readSint16BE();
- _game.cursorChar = in->readByte();
- _game.colorFg = in->readSint16BE();
- _game.colorBg = in->readSint16BE();
+ _text->inputSetCursorChar(in->readByte());
- // game.hires - rebuilt from image stack
- // game.sbuf - rebuilt from image stack
+ int16 textForeground = in->readSint16BE();
+ int16 textBackground = in->readSint16BE();
+ _text->charAttrib_Set(textForeground, textBackground);
// game.ego_words - fixed by clean_input
// game.num_ego_words - fixed by clean_input
@@ -407,41 +538,58 @@ int AgiEngine::loadGame(const Common::String &fileName, bool checkId) {
objectSetLocation(i, in->readSint16BE());
// Those are not serialized
- for (i = 0; i < MAX_DIRS; i++) {
+ for (i = 0; i < MAX_CONTROLLERS; i++) {
_game.controllerOccured[i] = false;
}
+ if (saveVersion >= 7) {
+ // For old saves, we just keep the current controllers
+ for (i = 0; i < MAX_CONTROLLER_KEYMAPPINGS; i++) {
+ _game.controllerKeyMapping[i].keycode = in->readUint16BE();
+ _game.controllerKeyMapping[i].controllerSlot = in->readByte();
+ }
+ }
+
+ if (saveVersion >= 8) {
+ // Version 8+: hold-key-mode
+ if (in->readByte()) {
+ _keyHoldMode = true;
+ } else {
+ _keyHoldMode = false;
+ }
+ }
+
for (i = 0; i < MAX_STRINGS; i++)
in->read(_game.strings[i], MAX_STRINGLEN);
- for (i = 0; i < MAX_DIRS; i++) {
+ for (i = 0; i < MAX_DIRECTORY_ENTRIES; i++) {
if (in->readByte() & RES_LOADED)
- agiLoadResource(rLOGIC, i);
+ agiLoadResource(RESOURCETYPE_LOGIC, i);
else
- agiUnloadResource(rLOGIC, i);
+ agiUnloadResource(RESOURCETYPE_LOGIC, i);
_game.logics[i].sIP = in->readSint16BE();
_game.logics[i].cIP = in->readSint16BE();
}
- for (i = 0; i < MAX_DIRS; i++) {
+ for (i = 0; i < MAX_DIRECTORY_ENTRIES; i++) {
if (in->readByte() & RES_LOADED)
- agiLoadResource(rPICTURE, i);
+ agiLoadResource(RESOURCETYPE_PICTURE, i);
else
- agiUnloadResource(rPICTURE, i);
+ agiUnloadResource(RESOURCETYPE_PICTURE, i);
}
- for (i = 0; i < MAX_DIRS; i++) {
+ for (i = 0; i < MAX_DIRECTORY_ENTRIES; i++) {
if (in->readByte() & RES_LOADED)
- agiLoadResource(rVIEW, i);
+ agiLoadResource(RESOURCETYPE_VIEW, i);
else
- agiUnloadResource(rVIEW, i);
+ agiUnloadResource(RESOURCETYPE_VIEW, i);
}
- for (i = 0; i < MAX_DIRS; i++) {
+ for (i = 0; i < MAX_DIRECTORY_ENTRIES; i++) {
if (in->readByte() & RES_LOADED)
- agiLoadResource(rSOUND, i);
+ agiLoadResource(RESOURCETYPE_SOUND, i);
else
- agiUnloadResource(rSOUND, i);
+ agiUnloadResource(RESOURCETYPE_SOUND, i);
}
// game.pictures - loaded above
@@ -450,78 +598,102 @@ int AgiEngine::loadGame(const Common::String &fileName, bool checkId) {
// game.sounds - loaded above
for (i = 0; i < vtEntries; i++) {
- VtEntry *v = &_game.viewTable[i];
-
- v->stepTime = in->readByte();
- v->stepTimeCount = in->readByte();
- v->entry = in->readByte();
- v->xPos = in->readSint16BE();
- v->yPos = in->readSint16BE();
- v->currentView = in->readByte();
-
- // v->view_data - fixed below
-
- v->currentLoop = in->readByte();
- v->numLoops = in->readByte();
-
- // v->loop_data - fixed below
-
- v->currentCel = in->readByte();
- v->numCels = in->readByte();
-
- // v->cel_data - fixed below
- // v->cel_data_2 - fixed below
-
- v->xPos2 = in->readSint16BE();
- v->yPos2 = in->readSint16BE();
-
- // v->s - fixed below
-
- v->xSize = in->readSint16BE();
- v->ySize = in->readSint16BE();
- v->stepSize = in->readByte();
- v->cycleTime = in->readByte();
- v->cycleTimeCount = in->readByte();
- v->direction = in->readByte();
-
- v->motion = (MotionType)in->readByte();
- v->cycle = (CycleType)in->readByte();
- v->priority = in->readByte();
-
- v->flags = in->readUint16BE();
-
- v->parm1 = in->readByte();
- v->parm2 = in->readByte();
- v->parm3 = in->readByte();
- v->parm4 = in->readByte();
+ ScreenObjEntry *screenObj = &_game.screenObjTable[i];
+
+ screenObj->stepTime = in->readByte();
+ screenObj->stepTimeCount = in->readByte();
+ screenObj->objectNr = in->readByte();
+ screenObj->xPos = in->readSint16BE();
+ screenObj->yPos = in->readSint16BE();
+ screenObj->currentViewNr = in->readByte();
+
+ // screenObj->view_data - fixed below
+
+ screenObj->currentLoopNr = in->readByte();
+ screenObj->loopCount = in->readByte();
+
+ // screenObj->loop_data - fixed below
+
+ screenObj->currentCelNr = in->readByte();
+ screenObj->celCount = in->readByte();
+
+ // screenObj->cel_data - fixed below
+ // screenObj->cel_data_2 - fixed below
+
+ screenObj->xPos_prev = in->readSint16BE();
+ screenObj->yPos_prev = in->readSint16BE();
+
+ // screenObj->s - fixed below
+
+ screenObj->xSize = in->readSint16BE();
+ screenObj->ySize = in->readSint16BE();
+ screenObj->stepSize = in->readByte();
+ screenObj->cycleTime = in->readByte();
+ screenObj->cycleTimeCount = in->readByte();
+ screenObj->direction = in->readByte();
+
+ screenObj->motionType = (MotionType)in->readByte();
+ screenObj->cycle = (CycleType)in->readByte();
+ screenObj->priority = in->readByte();
+
+ screenObj->flags = in->readUint16BE();
+
+ // this was done so that saved games compatibility isn't broken
+ switch (screenObj->motionType) {
+ case kMotionNormal:
+ in->readByte();
+ in->readByte();
+ in->readByte();
+ in->readByte();
+ break;
+ case kMotionWander:
+ screenObj->wander_count = in->readByte();
+ in->readByte();
+ in->readByte();
+ in->readByte();
+ break;
+ case kMotionFollowEgo:
+ screenObj->follow_stepSize = in->readByte();
+ screenObj->follow_flag = in->readByte();
+ screenObj->follow_count = in->readByte();
+ in->readByte();
+ break;
+ case kMotionEgo:
+ case kMotionMoveObj:
+ screenObj->move_x = in->readByte(); // problematic! int16 -> byte
+ screenObj->move_y = in->readByte();
+ screenObj->move_stepSize = in->readByte();
+ screenObj->move_flag = in->readByte();
+ break;
+ default:
+ error("unknown motion-type");
+ }
}
- for (i = vtEntries; i < MAX_VIEWTABLE; i++) {
- memset(&_game.viewTable[i], 0, sizeof(VtEntry));
+ for (i = vtEntries; i < SCREENOBJECTS_MAX; i++) {
+ memset(&_game.screenObjTable[i], 0, sizeof(ScreenObjEntry));
}
- // Fix some pointers in viewtable
+ // Fix some pointers in screenObjTable
- for (i = 0; i < MAX_VIEWTABLE; i++) {
- VtEntry *v = &_game.viewTable[i];
+ for (i = 0; i < SCREENOBJECTS_MAX; i++) {
+ ScreenObjEntry *screenObj = &_game.screenObjTable[i];
- if (_game.dirView[v->currentView].offset == _EMPTY)
+ if (_game.dirView[screenObj->currentViewNr].offset == _EMPTY)
continue;
- if (!(_game.dirView[v->currentView].flags & RES_LOADED))
- agiLoadResource(rVIEW, v->currentView);
+ if (!(_game.dirView[screenObj->currentViewNr].flags & RES_LOADED))
+ agiLoadResource(RESOURCETYPE_VIEW, screenObj->currentViewNr);
- setView(v, v->currentView); // Fix v->view_data
- setLoop(v, v->currentLoop); // Fix v->loop_data
- setCel(v, v->currentCel); // Fix v->cel_data
- v->celData2 = v->celData;
- v->s = NULL; // not sure if it is used...
+ setView(screenObj, screenObj->currentViewNr); // Fix v->view_data
+ setLoop(screenObj, screenObj->currentLoopNr); // Fix v->loop_data
+ setCel(screenObj, screenObj->currentCelNr); // Fix v->cel_data
}
- _sprites->eraseBoth();
+ _sprites->eraseSprites();
- // Clear input line
- _gfx->clearScreen(0);
- writeStatus();
+ _game.pictureShown = false;
+
+ _gfx->clearDisplay(0, false); // clear display screen, but not copy it to actual screen for now b/c transition
// Recreate background from saved image stack
clearImageStack();
@@ -529,7 +701,7 @@ int AgiEngine::loadGame(const Common::String &fileName, bool checkId) {
for (i = 0; i < 7; i++)
parm[i] = in->readSint16BE();
replayImageStackCall(t, parm[0], parm[1], parm[2],
- parm[3], parm[4], parm[5], parm[6]);
+ parm[3], parm[4], parm[5], parm[6]);
}
// Load AGIPAL Data
@@ -539,259 +711,28 @@ int AgiEngine::loadGame(const Common::String &fileName, bool checkId) {
delete in;
debugC(3, kDebugLevelMain | kDebugLevelSavegame, "Closed %s", fileName.c_str());
- setflag(fRestoreJustRan, true);
-
- _game.hasPrompt = 0; // force input line repaint if necessary
- cleanInput();
-
- _sprites->eraseBoth();
- _sprites->blitBoth();
- _sprites->commitBoth();
- _picture->showPic();
- _gfx->doUpdate();
-
- return errOK;
-}
-
-#define NUM_SLOTS 100
-#define NUM_VISIBLE_SLOTS 12
-
-Common::String AgiEngine::getSavegameFilename(int num) const {
- Common::String saveLoadSlot = _targetName;
- saveLoadSlot += Common::String::format(".%.3d", num);
- return saveLoadSlot;
-}
-
-void AgiEngine::getSavegameDescription(int num, char *buf, bool showEmpty) {
- Common::InSaveFile *in;
- Common::String fileName = getSavegameFilename(num);
-
- debugC(4, kDebugLevelMain | kDebugLevelSavegame, "Current game id is %s", _targetName.c_str());
-
- if (!(in = _saveFileMan->openForLoading(fileName))) {
- debugC(4, kDebugLevelMain | kDebugLevelSavegame, "File %s does not exist", fileName.c_str());
-
- if (showEmpty)
- strcpy(buf, " (empty slot)");
- else
- *buf = 0;
- } else {
- debugC(4, kDebugLevelMain | kDebugLevelSavegame, "Successfully opened %s for reading", fileName.c_str());
-
- uint32 type = in->readUint32BE();
-
- if (type == AGIflag) {
- debugC(6, kDebugLevelMain | kDebugLevelSavegame, "Has AGI flag, good start");
- in->read(buf, 31);
- } else {
- warning("This doesn't appear to be an AGI savegame");
- strcpy(buf, "(corrupt file)");
- }
-
- delete in;
- }
-}
-
-int AgiEngine::selectSlot() {
- int i, key, active = 0;
- int rc = -1;
- int hm = 1, vm = 3; // box margins
- int xmin, xmax, slotClicked;
- char desc[NUM_VISIBLE_SLOTS][40];
- int textCenter, buttonLength, buttonX[2], buttonY;
- const char *buttonText[] = { " OK ", "Cancel", NULL };
-
- _noSaveLoadAllowed = true;
-
- for (i = 0; i < NUM_VISIBLE_SLOTS; i++) {
- getSavegameDescription(_firstSlot + i, desc[i]);
- }
-
- textCenter = GFX_WIDTH / CHAR_LINES / 2;
- buttonLength = 6;
- buttonX[0] = (textCenter - 3 * buttonLength / 2) * CHAR_COLS;
- buttonX[1] = (textCenter + buttonLength / 2) * CHAR_COLS;
- buttonY = (vm + 17) * CHAR_LINES;
-
- for (i = 0; i < 2; i++)
- _gfx->drawCurrentStyleButton(buttonX[i], buttonY, buttonText[i], false, false, i == 0);
-
- AllowSyntheticEvents on(this);
- int oldFirstSlot = _firstSlot + 1;
- int oldActive = active + 1;
- bool exitSelectSlot = false;
- while (!exitSelectSlot && !(shouldQuit() || _restartGame)) {
- int sbPos = 0;
-
- // Use the extreme scrollbar positions only if the extreme
- // slots are in sight. (We have to calculate this even if we
- // don't redraw the save slots, because it's also used for
- // clicking in the scrollbar.
-
- if (_firstSlot == 0)
- sbPos = 1;
- else if (_firstSlot == NUM_SLOTS - NUM_VISIBLE_SLOTS)
- sbPos = NUM_VISIBLE_SLOTS - 2;
- else {
- sbPos = 2 + (_firstSlot * (NUM_VISIBLE_SLOTS - 4)) / (NUM_SLOTS - NUM_VISIBLE_SLOTS - 1);
- if (sbPos >= NUM_VISIBLE_SLOTS - 3)
- sbPos = NUM_VISIBLE_SLOTS - 3;
- }
-
- if (oldFirstSlot != _firstSlot || oldActive != active) {
- char dstr[64];
- for (i = 0; i < NUM_VISIBLE_SLOTS; i++) {
- sprintf(dstr, "[%2d. %-28.28s]", i + _firstSlot, desc[i]);
- printText(dstr, 0, hm + 1, vm + 4 + i,
- (40 - 2 * hm) - 1, i == active ? MSG_BOX_COLOR : MSG_BOX_TEXT,
- i == active ? MSG_BOX_TEXT : MSG_BOX_COLOR);
- }
+ setFlag(VM_FLAG_RESTORE_JUST_RAN, true);
- char upArrow[] = "^";
- char downArrow[] = "v";
- char scrollBar[] = " ";
+ _words->clearEgoWords();
- for (i = 1; i < NUM_VISIBLE_SLOTS - 1; i++)
- printText(scrollBar, 35, hm + 1, vm + 4 + i, 1, MSG_BOX_COLOR, 7, true);
-
- printText(upArrow, 35, hm + 1, vm + 4, 1, 8, 7);
- printText(downArrow, 35, hm + 1, vm + 4 + NUM_VISIBLE_SLOTS - 1, 1, 8, 7);
- printText(scrollBar, 35, hm + 1, vm + 4 + sbPos, 1, MSG_BOX_COLOR, MSG_BOX_TEXT);
-
- oldActive = active;
- oldFirstSlot = _firstSlot;
- }
+ // don't delay anything right after restoring a game
+ artificialDelay_Reset();
- pollTimer();
- key = doPollKeyboard();
-
- // It may happen that somebody will open GMM while
- // this dialog is open, and load a game
- // We are processing it here, effectively jumping
- // out of the dead loop
- if (getflag(fRestoreJustRan)) {
- rc = -2;
- exitSelectSlot = true;
- }
-
- if (!exitSelectSlot) {
- switch (key) {
- case KEY_ENTER:
- rc = active;
- Common::strlcpy(_game.strings[MAX_STRINGS], desc[i], MAX_STRINGLEN);
- debugC(8, kDebugLevelMain | kDebugLevelInput, "Button pressed: %d", rc);
- exitSelectSlot = true;
- break;
- case KEY_ESCAPE:
- rc = -1;
- exitSelectSlot = true;
- break;
- case BUTTON_LEFT:
- if (_gfx->testButton(buttonX[0], buttonY, buttonText[0])) {
- rc = active;
- strncpy(_game.strings[MAX_STRINGS], desc[i], MAX_STRINGLEN);
- debugC(8, kDebugLevelMain | kDebugLevelInput, "Button pressed: %d", rc);
- exitSelectSlot = true;
- } else if (_gfx->testButton(buttonX[1], buttonY, buttonText[1])) {
- rc = -1;
- exitSelectSlot = true;
- } else {
- slotClicked = ((int)_mouse.y - 1) / CHAR_COLS - (vm + 4);
- xmin = (hm + 1) * CHAR_COLS;
- xmax = xmin + CHAR_COLS * 34;
- if ((int)_mouse.x >= xmin && (int)_mouse.x <= xmax) {
- if (slotClicked >= 0 && slotClicked < NUM_VISIBLE_SLOTS)
- active = slotClicked;
- }
- xmin = (hm + 36) * CHAR_COLS;
- xmax = xmin + CHAR_COLS;
- if ((int)_mouse.x >= xmin && (int)_mouse.x <= xmax) {
- if (slotClicked >= 0 && slotClicked < NUM_VISIBLE_SLOTS) {
- if (slotClicked == 0)
- keyEnqueue(KEY_UP);
- else if (slotClicked == NUM_VISIBLE_SLOTS - 1)
- keyEnqueue(KEY_DOWN);
- else if (slotClicked < sbPos)
- keyEnqueue(KEY_UP_RIGHT);
- else if (slotClicked > sbPos)
- keyEnqueue(KEY_DOWN_RIGHT);
- }
- }
- }
- break;
-
- case KEY_DOWN:
- active++;
- if (active >= NUM_VISIBLE_SLOTS) {
- if (_firstSlot + NUM_VISIBLE_SLOTS < NUM_SLOTS) {
- _firstSlot++;
- for (i = 1; i < NUM_VISIBLE_SLOTS; i++)
- memcpy(desc[i - 1], desc[i], sizeof(desc[0]));
- getSavegameDescription(_firstSlot + NUM_VISIBLE_SLOTS - 1, desc[NUM_VISIBLE_SLOTS - 1]);
- }
- active = NUM_VISIBLE_SLOTS - 1;
- }
- break;
- case KEY_UP:
- active--;
- if (active < 0) {
- active = 0;
- if (_firstSlot > 0) {
- _firstSlot--;
- for (i = NUM_VISIBLE_SLOTS - 1; i > 0; i--)
- memcpy(desc[i], desc[i - 1], sizeof(desc[0]));
- getSavegameDescription(_firstSlot, desc[0]);
- }
- }
- break;
-
- // Page Up/Down and mouse wheel scrolling all leave 'active'
- // unchanged so that a visible slot will remain selected.
-
- case WHEEL_DOWN:
- if (_firstSlot < NUM_SLOTS - NUM_VISIBLE_SLOTS) {
- _firstSlot++;
- for (i = 1; i < NUM_VISIBLE_SLOTS; i++)
- memcpy(desc[i - 1], desc[i], sizeof(desc[0]));
- getSavegameDescription(_firstSlot + NUM_VISIBLE_SLOTS - 1, desc[NUM_VISIBLE_SLOTS - 1]);
- }
- break;
- case WHEEL_UP:
- if (_firstSlot > 0) {
- _firstSlot--;
- for (i = NUM_VISIBLE_SLOTS - 1; i > 0; i--)
- memcpy(desc[i], desc[i - 1], sizeof(desc[0]));
- getSavegameDescription(_firstSlot, desc[0]);
- }
- break;
- case KEY_DOWN_RIGHT:
- // This is probably triggered by Page Down.
- _firstSlot += NUM_VISIBLE_SLOTS;
- if (_firstSlot > NUM_SLOTS - NUM_VISIBLE_SLOTS) {
- _firstSlot = NUM_SLOTS - NUM_VISIBLE_SLOTS;
- }
- for (i = 0; i < NUM_VISIBLE_SLOTS; i++)
- getSavegameDescription(_firstSlot + i, desc[i]);
- break;
- case KEY_UP_RIGHT:
- // This is probably triggered by Page Up.
- _firstSlot -= NUM_VISIBLE_SLOTS;
- if (_firstSlot < 0) {
- _firstSlot = 0;
- }
- for (i = 0; i < NUM_VISIBLE_SLOTS; i++)
- getSavegameDescription(_firstSlot + i, desc[i]);
- break;
- }
- }
- _gfx->doUpdate();
- }
+ _sprites->eraseSprites();
+ _sprites->buildAllSpriteLists();
+ _sprites->drawAllSpriteLists();
+ _picture->showPicWithTransition();
+ _game.pictureShown = true;
+ _text->statusDraw();
+ _text->promptRedraw();
- closeWindow();
+ // copy everything over (we should probably only copy over the remaining parts of the screen w/o play screen
+ _gfx->copyDisplayToScreen();
- _noSaveLoadAllowed = false;
+ // Sync volume settings from ScummVM system settings, so that VM volume variable is overwritten
+ setVolumeViaSystemSetting();
- return rc;
+ return errOK;
}
int AgiEngine::scummVMSaveLoadDialog(bool isSave) {
@@ -834,7 +775,8 @@ int AgiEngine::doSave(int slot, const Common::String &desc) {
// Make sure all graphics was blitted to screen. This fixes bug
// #2960567: "AGI: Ego partly erased in Load/Save thumbnails"
- _gfx->doUpdate();
+ _gfx->updateScreen();
+// _gfx->doUpdate();
return saveGame(fileName, desc);
}
@@ -843,162 +785,218 @@ int AgiEngine::doLoad(int slot, bool showMessages) {
Common::String fileName = getSavegameFilename(slot);
debugC(8, kDebugLevelMain | kDebugLevelResources, "file is [%s]", fileName.c_str());
- _sprites->eraseBoth();
+ _sprites->eraseSprites();
_sound->stopSound();
- closeWindow();
+ _text->closeWindow();
int result = loadGame(fileName);
if (result == errOK) {
- if (showMessages)
- messageBox("Game restored.");
- _game.exitAllLogics = 1;
- _menu->enableAll();
+ _game.exitAllLogics = true;
+ _menu->itemEnableAll();
} else {
if (showMessages)
- messageBox("Error restoring game.");
+ _text->messageBox("Error restoring game.");
}
return result;
}
-int AgiEngine::saveGameDialog() {
- if (!ConfMan.getBool("originalsaveload"))
- return scummVMSaveLoadDialog(true);
+SavedGameSlotIdArray AgiEngine::getSavegameSlotIds() {
+ Common::StringArray filenames;
+ int16 numberPos = _targetName.size() + 1;
+ int16 slotId = 0;
+ SavedGameSlotIdArray slotIdArray;
- char *desc;
- const char *buttons[] = { "Do as I say!", "I regret", NULL };
- char dstr[200];
- int rc, slot = 0;
- int hm, vm, hp, vp;
- int w;
-
- hm = 1;
- vm = 3;
- hp = hm * CHAR_COLS;
- vp = vm * CHAR_LINES;
- w = (40 - 2 * hm) - 1;
-
- do {
- drawWindow(hp, vp, GFX_WIDTH - hp, GFX_HEIGHT - vp);
- printText("Select a slot in which you wish to\nsave the game:",
- 0, hm + 1, vm + 1, w, MSG_BOX_TEXT, MSG_BOX_COLOR);
- slot = selectSlot();
- if (slot + _firstSlot == 0)
- messageBox("That slot is for Autosave only.");
- else if (slot < 0)
- return errOK;
- } while (slot + _firstSlot == 0);
-
- drawWindow(hp, vp + 5 * CHAR_LINES, GFX_WIDTH - hp,
- GFX_HEIGHT - vp - 9 * CHAR_LINES);
- printText("Enter a description for this game:",
- 0, hm + 1, vm + 6, w, MSG_BOX_TEXT, MSG_BOX_COLOR);
- _gfx->drawRectangle(3 * CHAR_COLS, 11 * CHAR_LINES - 1,
- 37 * CHAR_COLS, 12 * CHAR_LINES, MSG_BOX_TEXT);
- _gfx->flushBlock(3 * CHAR_COLS, 11 * CHAR_LINES - 1,
- 37 * CHAR_COLS, 12 * CHAR_LINES);
-
- // The description field of the save/restore dialog holds 32 characters
- // but we use four of them for the slot number. The input field is a
- // bit wider than that, so we don't have to worry about leaving space
- // for the cursor.
-
- getString(2, 11, 28, MAX_STRINGS);
-
- // If we're saving over an old slot, show the old description. We can't
- // access that buffer directly, so we have to feed the characters to
- // the input handler one at a time.
-
- char name[40];
- int numChars;
-
- getSavegameDescription(_firstSlot + slot, name, false);
-
- for (numChars = 0; numChars < 28 && name[numChars]; numChars++)
- handleGetstring(name[numChars]);
-
- _gfx->printCharacter(numChars + 3, 11, _game.cursorChar, MSG_BOX_COLOR, MSG_BOX_TEXT);
- do {
- mainCycle();
- } while (_game.inputMode == INPUT_GETSTRING);
- closeWindow();
-
- desc = _game.strings[MAX_STRINGS];
- sprintf(dstr, "Are you sure you want to save the game "
- "described as:\n\n%s\n\nin slot %d?\n\n\n", desc, _firstSlot + slot);
-
- rc = selectionBox(dstr, buttons);
-
- if (rc != 0) {
- messageBox("Game NOT saved.");
- return errOK;
+ // search for saved game filenames...
+ filenames = _saveFileMan->listSavefiles(_targetName + ".###");
+
+ Common::StringArray::iterator it;
+ Common::StringArray::iterator end = filenames.end();;
+
+ // convert to lower-case, just to be sure
+ for (it = filenames.begin(); it != end; it++) {
+ it->toLowercase();
}
+ // sort
+ Common::sort(filenames.begin(), filenames.end());
- int result = doSave(_firstSlot + slot, desc);
+ // now extract slot-Ids
+ for (it = filenames.begin(); it != end; it++) {
+ slotId = atoi(it->c_str() + numberPos);
- if (result == errOK)
- messageBox("Game saved.");
- else
- messageBox("Error saving game.");
+ slotIdArray.push_back(slotId);
+ }
+ return slotIdArray;
+}
- return result;
+Common::String AgiEngine::getSavegameFilename(int16 slotId) const {
+ Common::String saveLoadSlot = _targetName;
+ saveLoadSlot += Common::String::format(".%.3d", slotId);
+ return saveLoadSlot;
}
-int AgiEngine::saveGameSimple() {
- if (!ConfMan.getBool("originalsaveload"))
- return scummVMSaveLoadDialog(true);
+bool AgiEngine::getSavegameInformation(int16 slotId, Common::String &saveDescription, uint32 &saveDate, uint32 &saveTime, bool &saveIsValid) {
+ Common::InSaveFile *in;
+ Common::String fileName = getSavegameFilename(slotId);
+ char saveGameDescription[31];
+ int16 curPos = 0;
+ byte saveVersion = 0;
- Common::String fileName = getSavegameFilename(0);
+ saveDescription.clear();
+ saveDate = 0;
+ saveTime = 0;
+ saveIsValid = false;
- int result = saveGame(fileName, "Default savegame");
- if (result != errOK)
- messageBox("Error saving game.");
- return result;
-}
+ debugC(4, kDebugLevelMain | kDebugLevelSavegame, "Current game id is %s", _targetName.c_str());
-int AgiEngine::loadGameDialog() {
- if (!ConfMan.getBool("originalsaveload"))
- return scummVMSaveLoadDialog(false);
+ if (!(in = _saveFileMan->openForLoading(fileName))) {
+ debugC(4, kDebugLevelMain | kDebugLevelSavegame, "File %s does not exist", fileName.c_str());
+ return false;
- int slot = 0;
- int hm, vm, hp, vp; // box margins
- int w;
+ } else {
+ debugC(4, kDebugLevelMain | kDebugLevelSavegame, "Successfully opened %s for reading", fileName.c_str());
- hm = 1;
- vm = 3;
- hp = hm * CHAR_COLS;
- vp = vm * CHAR_LINES;
- w = (40 - 2 * hm) - 1;
+ uint32 type = in->readUint32BE();
- _sprites->eraseBoth();
- _sound->stopSound();
+ if (type != AGIflag) {
+ warning("This doesn't appear to be an AGI savegame");
+ saveDescription += "[ScummVM: not an AGI save]";
+ delete in;
+ return true;
+ }
- drawWindow(hp, vp, GFX_WIDTH - hp, GFX_HEIGHT - vp);
- printText("Select a game which you wish to\nrestore:",
- 0, hm + 1, vm + 1, w, MSG_BOX_TEXT, MSG_BOX_COLOR);
+ debugC(6, kDebugLevelMain | kDebugLevelSavegame, "Has AGI flag, good start");
+ if (in->read(saveGameDescription, 31) != 31) {
+ warning("unexpected EOF");
+ delete in;
+ saveDescription += "[ScummVM: invalid save]";
+ return true;
+ }
- slot = selectSlot();
+ for (curPos = 0; curPos < 31; curPos++) {
+ if (!saveGameDescription[curPos])
+ break;
+ }
+ if (curPos >= 31) {
+ warning("corrupted description");
+ delete in;
+ saveDescription += "[ScummVM: invalid save]";
+ return true;
+ }
- if (slot < 0) {
- if (slot == -1) // slot = -2 when GMM was launched
- messageBox("Game NOT restored.");
+ saveVersion = in->readByte();
+ if (saveVersion > SAVEGAME_CURRENT_VERSION) {
+ warning("save from a future ScummVM, not supported");
+ delete in;
+ saveDescription += "[ScummVM: not supported]";
+ return true;
+ }
- return errOK;
+ if (saveVersion >= 4) {
+ // We don't need the thumbnail here, so just read it and discard it
+ Graphics::skipThumbnail(*in);
+
+ saveDate = in->readUint32BE();
+ saveTime = in->readUint16BE() << 8;
+ if (saveVersion >= 9) {
+ saveTime |= in->readByte(); // add seconds (only available since saved game version 9+)
+ }
+
+ // save date is DDMMYYYY, we need a proper format
+ byte saveDateDay = saveDate >> 24;
+ byte saveDateMonth = (saveDate >> 16) & 0xFF;
+ uint16 saveDateYear = saveDate & 0xFFFF;
+
+ saveDate = (saveDateYear << 16) | (saveDateMonth << 8) | saveDateDay;
+
+ } else {
+ saveDate = 0;
+ saveTime = 0;
+ }
+
+ saveDescription += saveGameDescription;
+ saveIsValid = true;
+
+ delete in;
+ return true;
}
+}
- return doLoad(_firstSlot + slot, true);
+bool AgiEngine::loadGameAutomatic() {
+ int16 automaticRestoreGameSlotId = 0;
+
+ automaticRestoreGameSlotId = _systemUI->figureOutAutomaticRestoreGameSlot(_game.automaticSaveDescription);
+ if (automaticRestoreGameSlotId >= 0) {
+ if (doLoad(automaticRestoreGameSlotId, true) == errOK) {
+ return true;
+ }
+ }
+ return false;
}
-int AgiEngine::loadGameSimple() {
+bool AgiEngine::loadGameDialog() {
+ int16 restoreGameSlotId = 0;
+
if (!ConfMan.getBool("originalsaveload"))
return scummVMSaveLoadDialog(false);
- else
- return doLoad(0, true);
+
+ restoreGameSlotId = _systemUI->askForRestoreGameSlot();
+ if (restoreGameSlotId >= 0) {
+ if (doLoad(restoreGameSlotId, true) == errOK) {
+ return true;
+ }
+ }
+ return errOK;
+}
+
+// Try to figure out either the slot, that is currently using the automatic saved game description
+// or get a new slot.
+// If we fail, return false, so that the regular saved game dialog is called
+// Original AGI was limited to 12 saves, we are effectively limited to 100 saves at the moment.
+//
+// btw. this also means that entering an existant name in Mixed Up Mother Goose will effectively overwrite
+// that saved game. This is also what original AGI did.
+bool AgiEngine::saveGameAutomatic() {
+ int16 automaticSaveGameSlotId = 0;
+
+ automaticSaveGameSlotId = _systemUI->figureOutAutomaticSaveGameSlot(_game.automaticSaveDescription);
+ if (automaticSaveGameSlotId >= 0) {
+ Common::String slotDescription(_game.automaticSaveDescription);
+
+ // WORKAROUND: Remove window in case one is currently shown, otherwise it would get saved in the thumbnail
+ // Happens for Mixed Up Mother Goose. The scripts close the window after saving.
+ // Original interpreter obviously did not do this, but original interpreter also did not save thumbnails.
+ _text->closeWindow();
+
+ if (doSave(automaticSaveGameSlotId, slotDescription) == errOK) {
+ return true;
+ }
+ }
+ return false;
+}
+
+bool AgiEngine::saveGameDialog() {
+ int16 saveGameSlotId = 0;
+ Common::String slotDescription;
+
+ if (!ConfMan.getBool("originalsaveload"))
+ return scummVMSaveLoadDialog(true);
+
+ saveGameSlotId = _systemUI->askForSaveGameSlot();
+ if (saveGameSlotId >= 0) {
+ if (_systemUI->askForSaveGameDescription(saveGameSlotId, slotDescription)) {
+ if (doSave(saveGameSlotId, slotDescription) == errOK) {
+ return true;
+ }
+ }
+ }
+ return false;
}
+
void AgiEngine::recordImageStackCall(uint8 type, int16 p1, int16 p2, int16 p3,
- int16 p4, int16 p5, int16 p6, int16 p7) {
+ int16 p4, int16 p5, int16 p6, int16 p7) {
ImageStackElement pnew;
pnew.type = type;
@@ -1015,15 +1013,15 @@ void AgiEngine::recordImageStackCall(uint8 type, int16 p1, int16 p2, int16 p3,
}
void AgiEngine::replayImageStackCall(uint8 type, int16 p1, int16 p2, int16 p3,
- int16 p4, int16 p5, int16 p6, int16 p7) {
+ int16 p4, int16 p5, int16 p6, int16 p7) {
switch (type) {
case ADD_PIC:
debugC(8, kDebugLevelMain, "--- decoding picture %d ---", p1);
- agiLoadResource(rPICTURE, p1);
+ agiLoadResource(RESOURCETYPE_PICTURE, p1);
_picture->decodePicture(p1, p2, p3 != 0);
break;
case ADD_VIEW:
- agiLoadResource(rVIEW, p1);
+ agiLoadResource(RESOURCETYPE_VIEW, p1);
_sprites->addToPic(p1, p2, p3, p4, p5, p6, p7);
break;
}
@@ -1041,12 +1039,12 @@ void AgiEngine::checkQuickLoad() {
if (ConfMan.hasKey("save_slot")) {
Common::String saveNameBuffer = getSavegameFilename(ConfMan.getInt("save_slot"));
- _sprites->eraseBoth();
+ _sprites->eraseSprites();
_sound->stopSound();
- if (loadGame(saveNameBuffer, false) == errOK) { // Do not check game id
- _game.exitAllLogics = 1;
- _menu->enableAll();
+ if (loadGame(saveNameBuffer, false) == errOK) { // Do not check game id
+ _game.exitAllLogics = true;
+ _menu->itemEnableAll();
}
}
}
@@ -1054,21 +1052,21 @@ void AgiEngine::checkQuickLoad() {
Common::Error AgiEngine::loadGameState(int slot) {
Common::String saveLoadSlot = getSavegameFilename(slot);
- _sprites->eraseBoth();
+ _sprites->eraseSprites();
_sound->stopSound();
if (loadGame(saveLoadSlot) == errOK) {
- _game.exitAllLogics = 1;
- _menu->enableAll();
+ _game.exitAllLogics = true;
+ _menu->itemEnableAll();
return Common::kNoError;
} else {
return Common::kUnknownError;
}
}
-Common::Error AgiEngine::saveGameState(int slot, const Common::String &desc) {
+Common::Error AgiEngine::saveGameState(int slot, const Common::String &description) {
Common::String saveLoadSlot = getSavegameFilename(slot);
- if (saveGame(saveLoadSlot, desc) == errOK)
+ if (saveGame(saveLoadSlot, description) == errOK)
return Common::kNoError;
else
return Common::kUnknownError;