2015-01-02 05:15:08 +00:00
|
|
|
/* 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 "common/scummsys.h"
|
|
|
|
#include "common/algorithm.h"
|
2015-01-06 02:17:44 +00:00
|
|
|
#include "common/memstream.h"
|
2018-01-12 02:31:48 +00:00
|
|
|
#include "common/substream.h"
|
2018-01-28 02:55:26 +00:00
|
|
|
#include "common/translation.h"
|
2017-12-22 13:52:31 +00:00
|
|
|
#include "graphics/scaler.h"
|
|
|
|
#include "graphics/thumbnail.h"
|
2018-01-28 02:55:26 +00:00
|
|
|
#include "gui/saveload.h"
|
2015-01-02 05:15:08 +00:00
|
|
|
#include "xeen/saves.h"
|
2015-01-02 21:01:41 +00:00
|
|
|
#include "xeen/files.h"
|
|
|
|
#include "xeen/xeen.h"
|
2015-01-02 05:15:08 +00:00
|
|
|
|
|
|
|
namespace Xeen {
|
|
|
|
|
2017-12-22 13:52:31 +00:00
|
|
|
SavesManager::SavesManager(const Common::String &targetName): _targetName(targetName),
|
|
|
|
_wonWorld(false), _wonDarkSide(false) {
|
|
|
|
File::_xeenSave = nullptr;
|
|
|
|
File::_darkSave = nullptr;
|
|
|
|
}
|
2015-01-11 19:21:57 +00:00
|
|
|
|
2017-12-22 13:52:31 +00:00
|
|
|
SavesManager::~SavesManager() {
|
|
|
|
delete File::_xeenSave;
|
|
|
|
delete File::_darkSave;
|
|
|
|
}
|
2015-01-11 19:21:57 +00:00
|
|
|
|
2017-12-22 13:52:31 +00:00
|
|
|
const char *const SAVEGAME_STR = "XEEN";
|
|
|
|
#define SAVEGAME_STR_SIZE 6
|
|
|
|
|
|
|
|
bool SavesManager::readSavegameHeader(Common::InSaveFile *in, XeenSavegameHeader &header) {
|
|
|
|
char saveIdentBuffer[SAVEGAME_STR_SIZE + 1];
|
|
|
|
header._thumbnail = nullptr;
|
|
|
|
|
|
|
|
// Validate the header Id
|
|
|
|
in->read(saveIdentBuffer, SAVEGAME_STR_SIZE + 1);
|
|
|
|
if (strncmp(saveIdentBuffer, SAVEGAME_STR, SAVEGAME_STR_SIZE))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
header._version = in->readByte();
|
|
|
|
if (header._version > XEEN_SAVEGAME_VERSION)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Read in the string
|
|
|
|
header._saveName.clear();
|
|
|
|
char ch;
|
|
|
|
while ((ch = (char)in->readByte()) != '\0')
|
|
|
|
header._saveName += ch;
|
|
|
|
|
|
|
|
// Get the thumbnail
|
|
|
|
header._thumbnail = Graphics::loadThumbnail(*in);
|
|
|
|
if (!header._thumbnail)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Read in save date/time
|
|
|
|
header._year = in->readSint16LE();
|
|
|
|
header._month = in->readSint16LE();
|
|
|
|
header._day = in->readSint16LE();
|
|
|
|
header._hour = in->readSint16LE();
|
|
|
|
header._minute = in->readSint16LE();
|
|
|
|
header._totalFrames = in->readUint32LE();
|
|
|
|
|
|
|
|
return true;
|
2015-01-02 05:15:08 +00:00
|
|
|
}
|
|
|
|
|
2017-12-22 13:52:31 +00:00
|
|
|
void SavesManager::writeSavegameHeader(Common::OutSaveFile *out, XeenSavegameHeader &header) {
|
2018-01-12 02:31:48 +00:00
|
|
|
EventsManager &events = *g_vm->_events;
|
|
|
|
Screen &screen = *g_vm->_screen;
|
|
|
|
|
2017-12-22 13:52:31 +00:00
|
|
|
// Write out a savegame header
|
|
|
|
out->write(SAVEGAME_STR, SAVEGAME_STR_SIZE + 1);
|
|
|
|
|
|
|
|
out->writeByte(XEEN_SAVEGAME_VERSION);
|
|
|
|
|
|
|
|
// Write savegame name
|
|
|
|
out->writeString(header._saveName);
|
|
|
|
out->writeByte('\0');
|
|
|
|
|
|
|
|
// Write a thumbnail of the screen
|
|
|
|
uint8 thumbPalette[768];
|
2018-01-12 02:31:48 +00:00
|
|
|
screen.getPalette(thumbPalette);
|
2017-12-22 13:52:31 +00:00
|
|
|
Graphics::Surface saveThumb;
|
2018-01-12 02:31:48 +00:00
|
|
|
::createThumbnail(&saveThumb, (const byte *)screen.getPixels(),
|
|
|
|
screen.w, screen.h, thumbPalette);
|
2017-12-22 13:52:31 +00:00
|
|
|
Graphics::saveThumbnail(*out, saveThumb);
|
|
|
|
saveThumb.free();
|
2018-01-12 02:31:48 +00:00
|
|
|
|
2017-12-22 13:52:31 +00:00
|
|
|
// Write out the save date/time
|
|
|
|
TimeDate td;
|
|
|
|
g_system->getTimeAndDate(td);
|
|
|
|
out->writeSint16LE(td.tm_year + 1900);
|
|
|
|
out->writeSint16LE(td.tm_mon + 1);
|
|
|
|
out->writeSint16LE(td.tm_mday);
|
|
|
|
out->writeSint16LE(td.tm_hour);
|
|
|
|
out->writeSint16LE(td.tm_min);
|
2018-01-12 02:31:48 +00:00
|
|
|
out->writeUint32LE(events.playTime());
|
2015-01-02 05:15:08 +00:00
|
|
|
}
|
|
|
|
|
2017-12-22 13:52:31 +00:00
|
|
|
Common::Error SavesManager::saveGameState(int slot, const Common::String &desc) {
|
|
|
|
Common::OutSaveFile *out = g_system->getSavefileManager()->openForSaving(
|
|
|
|
generateSaveName(slot));
|
|
|
|
if (!out)
|
|
|
|
return Common::kCreatingFileFailed;
|
2015-01-06 02:17:44 +00:00
|
|
|
|
2018-01-13 02:22:06 +00:00
|
|
|
// Push map and party data to the save archives
|
|
|
|
Map &map = *g_vm->_map;
|
|
|
|
map.saveMaze();
|
|
|
|
|
2018-01-14 19:16:58 +00:00
|
|
|
// Write the savegame header
|
2017-12-22 13:52:31 +00:00
|
|
|
XeenSavegameHeader header;
|
|
|
|
header._saveName = desc;
|
|
|
|
writeSavegameHeader(out, header);
|
2015-01-11 19:21:57 +00:00
|
|
|
|
2018-01-12 02:31:48 +00:00
|
|
|
// Loop through saving the sides' save archives
|
|
|
|
SaveArchive *archives[2] = { File::_xeenSave, File::_darkSave };
|
|
|
|
for (int idx = 0; idx < 2; ++idx) {
|
|
|
|
if (archives[idx]) {
|
|
|
|
archives[idx]->save(*out);
|
|
|
|
} else {
|
|
|
|
// Side isn't present
|
|
|
|
out->writeUint32LE(0);
|
|
|
|
}
|
|
|
|
}
|
2015-01-06 02:17:44 +00:00
|
|
|
|
2018-01-13 22:29:40 +00:00
|
|
|
// Write out miscellaneous
|
|
|
|
FileManager &files = *g_vm->_files;
|
|
|
|
files.save(*out);
|
|
|
|
|
2017-12-22 13:52:31 +00:00
|
|
|
out->finalize();
|
|
|
|
delete out;
|
2015-01-06 02:17:44 +00:00
|
|
|
|
2017-12-22 13:52:31 +00:00
|
|
|
return Common::kNoError;
|
|
|
|
}
|
2015-01-06 02:17:44 +00:00
|
|
|
|
2017-12-22 13:52:31 +00:00
|
|
|
Common::Error SavesManager::loadGameState(int slot) {
|
2018-01-13 22:29:40 +00:00
|
|
|
EventsManager &events = *g_vm->_events;
|
|
|
|
FileManager &files = *g_vm->_files;
|
|
|
|
Map &map = *g_vm->_map;
|
|
|
|
Party &party = *g_vm->_party;
|
|
|
|
|
2017-12-22 13:52:31 +00:00
|
|
|
Common::InSaveFile *saveFile = g_system->getSavefileManager()->openForLoading(
|
|
|
|
generateSaveName(slot));
|
|
|
|
if (!saveFile)
|
|
|
|
return Common::kReadingFailed;
|
2015-01-06 02:17:44 +00:00
|
|
|
|
2017-12-22 13:52:31 +00:00
|
|
|
// Load the savaegame header
|
|
|
|
XeenSavegameHeader header;
|
|
|
|
if (!readSavegameHeader(saveFile, header))
|
|
|
|
error("Invalid savegame");
|
2015-01-02 05:15:08 +00:00
|
|
|
|
2017-12-22 13:52:31 +00:00
|
|
|
if (header._thumbnail) {
|
|
|
|
header._thumbnail->free();
|
|
|
|
delete header._thumbnail;
|
2015-01-08 03:11:18 +00:00
|
|
|
}
|
2015-01-06 02:17:44 +00:00
|
|
|
|
2018-01-12 02:31:48 +00:00
|
|
|
// Set the total play time
|
2018-01-13 22:29:40 +00:00
|
|
|
events.setPlayTime(header._totalFrames);
|
2018-01-12 02:31:48 +00:00
|
|
|
|
|
|
|
// Loop through loading the sides' save archives
|
|
|
|
SaveArchive *archives[2] = { File::_xeenSave, File::_darkSave };
|
|
|
|
for (int idx = 0; idx < 2; ++idx) {
|
|
|
|
uint fileSize = saveFile->readUint32LE();
|
|
|
|
|
|
|
|
if (archives[idx]) {
|
|
|
|
Common::SeekableSubReadStream arcStream(saveFile, saveFile->pos(),
|
|
|
|
saveFile->pos() + fileSize);
|
2018-01-13 02:22:06 +00:00
|
|
|
archives[idx]->load(arcStream);
|
2018-01-12 02:31:48 +00:00
|
|
|
} else {
|
|
|
|
assert(!fileSize);
|
|
|
|
}
|
|
|
|
}
|
2015-01-08 03:11:18 +00:00
|
|
|
|
2018-01-13 22:29:40 +00:00
|
|
|
// Read in miscellaneous
|
|
|
|
files.load(*saveFile);
|
|
|
|
|
|
|
|
// Load the new map
|
2018-02-24 21:14:59 +00:00
|
|
|
map.clearMaze();
|
2018-03-18 00:27:42 +00:00
|
|
|
map._loadDarkSide = files._ccNum;
|
2018-01-13 22:29:40 +00:00
|
|
|
map.load(party._mazeId);
|
|
|
|
|
2018-02-13 02:24:10 +00:00
|
|
|
delete saveFile;
|
2017-12-22 13:52:31 +00:00
|
|
|
return Common::kNoError;
|
2015-01-06 02:17:44 +00:00
|
|
|
}
|
|
|
|
|
2017-12-22 13:52:31 +00:00
|
|
|
Common::String SavesManager::generateSaveName(int slot) {
|
|
|
|
return Common::String::format("%s.%03d", _targetName.c_str(), slot);
|
2015-01-05 13:11:16 +00:00
|
|
|
}
|
|
|
|
|
2018-02-26 23:24:24 +00:00
|
|
|
void SavesManager::newGame() {
|
|
|
|
delete File::_xeenSave;
|
|
|
|
delete File::_darkSave;
|
|
|
|
File::_xeenSave = nullptr;
|
|
|
|
File::_darkSave = nullptr;
|
|
|
|
|
|
|
|
if (g_vm->getGameID() != GType_Clouds) {
|
|
|
|
File::_darkSave = new SaveArchive(g_vm->_party);
|
|
|
|
File::_darkSave->reset(File::_darkCc);
|
|
|
|
}
|
|
|
|
if (g_vm->getGameID() != GType_DarkSide && g_vm->getGameID() != GType_Swords) {
|
|
|
|
File::_xeenSave = new SaveArchive(g_vm->_party);
|
|
|
|
File::_xeenSave->reset(File::_xeenCc);
|
|
|
|
}
|
|
|
|
|
|
|
|
File::_currentSave = g_vm->getGameID() == GType_DarkSide || g_vm->getGameID() == GType_Swords ?
|
|
|
|
File::_darkSave : File::_xeenSave;
|
|
|
|
assert(File::_currentSave);
|
|
|
|
|
|
|
|
// Set any final initial values
|
|
|
|
Party &party = *g_vm->_party;
|
|
|
|
party.resetBlacksmithWares();
|
|
|
|
party._totalTime = 0;
|
|
|
|
|
|
|
|
switch (g_vm->getGameID()) {
|
|
|
|
case GType_Swords:
|
|
|
|
party._year = 1050;
|
|
|
|
break;
|
|
|
|
case GType_DarkSide:
|
|
|
|
party._year = 850;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
party._year = 610;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
party._day = 1;
|
|
|
|
}
|
|
|
|
|
2018-01-28 01:00:24 +00:00
|
|
|
bool SavesManager::loadGame() {
|
2018-01-28 02:55:26 +00:00
|
|
|
GUI::SaveLoadChooser *dialog = new GUI::SaveLoadChooser(_("Load game:"), _("Load"), false);
|
|
|
|
int slotNum = dialog->runModalWithCurrentTarget();
|
|
|
|
delete dialog;
|
|
|
|
|
|
|
|
if (slotNum != -1)
|
|
|
|
loadGameState(slotNum);
|
|
|
|
|
|
|
|
return slotNum != -1;
|
2018-01-28 01:00:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool SavesManager::saveGame() {
|
2018-01-28 02:55:26 +00:00
|
|
|
if (!g_vm->canSaveGameStateCurrently())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
GUI::SaveLoadChooser *dialog = new GUI::SaveLoadChooser(_("Save game:"), _("Save"), true);
|
|
|
|
int slotNum = dialog->runModalWithCurrentTarget();
|
|
|
|
Common::String saveName = dialog->getResultString();
|
|
|
|
delete dialog;
|
|
|
|
|
2018-03-13 23:04:08 +00:00
|
|
|
if (slotNum != -1)
|
2018-01-28 02:55:26 +00:00
|
|
|
saveGameState(slotNum, saveName);
|
|
|
|
|
|
|
|
return slotNum != -1;
|
2018-01-28 01:00:24 +00:00
|
|
|
}
|
|
|
|
|
2015-01-02 05:15:08 +00:00
|
|
|
} // End of namespace Xeen
|