2012-10-24 22:49:32 +11: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/system.h"
|
|
|
|
#include "common/savefile.h"
|
2012-10-27 20:19:21 +11:00
|
|
|
#include "graphics/surface.h"
|
|
|
|
#include "graphics/scaler.h"
|
|
|
|
#include "graphics/thumbnail.h"
|
2012-10-24 22:49:32 +11:00
|
|
|
#include "hopkins/saveload.h"
|
|
|
|
#include "hopkins/files.h"
|
|
|
|
#include "hopkins/globals.h"
|
2012-10-27 20:19:21 +11:00
|
|
|
#include "hopkins/graphics.h"
|
|
|
|
#include "hopkins/hopkins.h"
|
2012-10-24 22:49:32 +11:00
|
|
|
|
|
|
|
namespace Hopkins {
|
|
|
|
|
2012-10-27 20:19:21 +11:00
|
|
|
const char *SAVEGAME_STR = "HOPKINS";
|
|
|
|
#define SAVEGAME_STR_SIZE 13
|
|
|
|
|
2012-10-24 22:49:32 +11:00
|
|
|
void SaveLoadManager::setParent(HopkinsEngine *vm) {
|
|
|
|
_vm = vm;
|
|
|
|
}
|
|
|
|
|
2012-12-20 00:30:01 +01:00
|
|
|
bool SaveLoadManager::save(const Common::String &file, const void *buf, size_t n) {
|
2012-10-24 22:49:32 +11:00
|
|
|
Common::OutSaveFile *f = g_system->getSavefileManager()->openForSaving(file);
|
|
|
|
|
|
|
|
if (f) {
|
|
|
|
size_t bytesWritten = f->write(buf, n);
|
|
|
|
f->finalize();
|
|
|
|
delete f;
|
|
|
|
|
|
|
|
return bytesWritten == n;
|
2012-12-14 01:49:22 +01:00
|
|
|
} else
|
2012-10-24 22:49:32 +11:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save File
|
2012-12-20 00:30:01 +01:00
|
|
|
bool SaveLoadManager::saveFile(const Common::String &file, const void *buf, size_t n) {
|
|
|
|
return save(file, buf, n);
|
2012-10-24 22:49:32 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
void SaveLoadManager::initSaves() {
|
|
|
|
Common::String dataFilename = "HISCORE.DAT";
|
|
|
|
byte data[100];
|
|
|
|
Common::fill(&data[0], &data[100], 0);
|
2012-12-14 01:49:22 +01:00
|
|
|
|
2012-12-20 00:30:01 +01:00
|
|
|
saveFile(dataFilename, data, 100);
|
2012-10-24 22:49:32 +11:00
|
|
|
}
|
|
|
|
|
2012-12-20 00:30:01 +01:00
|
|
|
void SaveLoadManager::load(const Common::String &file, byte *buf) {
|
2012-10-24 23:20:26 +11:00
|
|
|
Common::InSaveFile *f = g_system->getSavefileManager()->openForLoading(file);
|
|
|
|
if (f == NULL)
|
|
|
|
error("Error openinig file - %s", file.c_str());
|
|
|
|
|
|
|
|
int32 filesize = f->size();
|
|
|
|
f->read(buf, filesize);
|
|
|
|
delete f;
|
|
|
|
}
|
|
|
|
|
2012-10-27 20:19:21 +11:00
|
|
|
bool SaveLoadManager::readSavegameHeader(Common::InSaveFile *in, hopkinsSavegameHeader &header) {
|
|
|
|
char saveIdentBuffer[SAVEGAME_STR_SIZE + 1];
|
2012-12-20 00:30:01 +01:00
|
|
|
header._thumbnail = NULL;
|
2012-10-27 20:19:21 +11:00
|
|
|
|
|
|
|
// Validate the header Id
|
|
|
|
in->read(saveIdentBuffer, SAVEGAME_STR_SIZE + 1);
|
|
|
|
if (strncmp(saveIdentBuffer, SAVEGAME_STR, SAVEGAME_STR_SIZE))
|
|
|
|
return false;
|
|
|
|
|
2012-12-20 00:30:01 +01:00
|
|
|
header._version = in->readByte();
|
|
|
|
if (header._version > HOPKINS_SAVEGAME_VERSION)
|
2012-10-27 20:19:21 +11:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// Read in the string
|
2012-12-20 00:30:01 +01:00
|
|
|
header._saveName.clear();
|
2012-10-27 20:19:21 +11:00
|
|
|
char ch;
|
2012-12-20 00:30:01 +01:00
|
|
|
while ((ch = (char)in->readByte()) != '\0') header._saveName += ch;
|
2012-10-27 20:19:21 +11:00
|
|
|
|
|
|
|
// Get the thumbnail
|
2012-12-20 00:30:01 +01:00
|
|
|
header._thumbnail = Graphics::loadThumbnail(*in);
|
|
|
|
if (!header._thumbnail)
|
2012-10-27 20:19:21 +11:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// Read in save date/time
|
2012-12-20 00:30:01 +01:00
|
|
|
header._year = in->readSint16LE();
|
|
|
|
header._month = in->readSint16LE();
|
|
|
|
header._day = in->readSint16LE();
|
|
|
|
header._hour = in->readSint16LE();
|
|
|
|
header._minute = in->readSint16LE();
|
|
|
|
header._totalFrames = in->readUint32LE();
|
2012-10-27 20:19:21 +11:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SaveLoadManager::writeSavegameHeader(Common::OutSaveFile *out, hopkinsSavegameHeader &header) {
|
|
|
|
// Write out a savegame header
|
|
|
|
out->write(SAVEGAME_STR, SAVEGAME_STR_SIZE + 1);
|
|
|
|
|
|
|
|
out->writeByte(HOPKINS_SAVEGAME_VERSION);
|
|
|
|
|
|
|
|
// Write savegame name
|
2012-12-20 00:30:01 +01:00
|
|
|
out->write(header._saveName.c_str(), header._saveName.size() + 1);
|
2012-10-27 20:19:21 +11:00
|
|
|
|
|
|
|
// Create a thumbnail and save it
|
|
|
|
Graphics::Surface *thumb = new Graphics::Surface();
|
2012-10-27 22:16:54 +11:00
|
|
|
createThumbnail(thumb);
|
2012-10-27 20:19:21 +11:00
|
|
|
Graphics::saveThumbnail(*out, *thumb);
|
|
|
|
thumb->free();
|
|
|
|
delete thumb;
|
|
|
|
|
|
|
|
// 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);
|
|
|
|
out->writeUint32LE(_vm->_eventsManager._gameCounter);
|
|
|
|
}
|
|
|
|
|
2012-12-20 00:30:01 +01:00
|
|
|
Common::Error SaveLoadManager::saveGame(int slot, const Common::String &saveName) {
|
2012-10-28 15:46:52 +11:00
|
|
|
/* Pack any necessary data into the savegame data structure */
|
|
|
|
// Set the selected slot number
|
|
|
|
_vm->_globals.SAUVEGARDE->data[svField10] = slot;
|
|
|
|
|
|
|
|
// Set up the inventory
|
|
|
|
for (int i = 0; i < 35; ++i)
|
2012-12-17 00:16:49 +01:00
|
|
|
_vm->_globals.SAUVEGARDE->_inventory[i] = _vm->_globals._inventory[i];
|
2012-10-28 15:46:52 +11:00
|
|
|
|
|
|
|
/* Create the savegame */
|
2012-10-27 22:16:54 +11:00
|
|
|
Common::OutSaveFile *saveFile = g_system->getSavefileManager()->openForSaving(
|
|
|
|
_vm->generateSaveName(slot));
|
|
|
|
if (!saveFile)
|
|
|
|
return Common::kCreatingFileFailed;
|
|
|
|
|
|
|
|
// Set up the serializer
|
|
|
|
Common::Serializer serializer(NULL, saveFile);
|
|
|
|
|
|
|
|
// Write out the savegame header
|
|
|
|
hopkinsSavegameHeader header;
|
2012-12-20 00:30:01 +01:00
|
|
|
header._saveName = saveName;
|
|
|
|
header._version = HOPKINS_SAVEGAME_VERSION;
|
2012-10-27 22:16:54 +11:00
|
|
|
writeSavegameHeader(saveFile, header);
|
|
|
|
|
|
|
|
// Write out the savegame data
|
|
|
|
syncSavegameData(serializer);
|
|
|
|
|
|
|
|
// Save file complete
|
|
|
|
saveFile->finalize();
|
|
|
|
delete saveFile;
|
|
|
|
|
|
|
|
return Common::kNoError;
|
|
|
|
}
|
|
|
|
|
2012-12-20 00:30:01 +01:00
|
|
|
Common::Error SaveLoadManager::loadGame(int slot) {
|
2012-10-27 22:16:54 +11:00
|
|
|
// Try and open the save file for reading
|
|
|
|
Common::InSaveFile *saveFile = g_system->getSavefileManager()->openForLoading(
|
|
|
|
_vm->generateSaveName(slot));
|
|
|
|
if (!saveFile)
|
|
|
|
return Common::kReadingFailed;
|
|
|
|
|
|
|
|
// Set up the serializer
|
|
|
|
Common::Serializer serializer(saveFile, NULL);
|
|
|
|
|
|
|
|
// Read in the savegame header
|
|
|
|
hopkinsSavegameHeader header;
|
|
|
|
readSavegameHeader(saveFile, header);
|
2012-12-20 00:30:01 +01:00
|
|
|
if (header._thumbnail)
|
|
|
|
header._thumbnail->free();
|
|
|
|
delete header._thumbnail;
|
2012-10-27 22:16:54 +11:00
|
|
|
|
|
|
|
// Read in the savegame data
|
|
|
|
syncSavegameData(serializer);
|
|
|
|
|
|
|
|
// Loading save file complete
|
|
|
|
delete saveFile;
|
|
|
|
|
2012-10-28 11:01:30 +11:00
|
|
|
// Unpack the inventory
|
2012-12-14 01:49:22 +01:00
|
|
|
for (int i = 0; i < 35; ++i)
|
2012-12-17 00:16:49 +01:00
|
|
|
_vm->_globals._inventory[i] = _vm->_globals.SAUVEGARDE->_inventory[i];
|
2012-10-28 11:01:30 +11:00
|
|
|
|
|
|
|
// Set variables from loaded data as necessary
|
|
|
|
_vm->_globals.SAUVEGARDE->data[svField10] = slot;
|
|
|
|
_vm->_globals.SORTIE = _vm->_globals.SAUVEGARDE->data[svField5];
|
|
|
|
_vm->_globals.SAUVEGARDE->data[svField6] = 0;
|
|
|
|
_vm->_globals.ECRAN = 0;
|
|
|
|
|
2012-10-27 22:16:54 +11:00
|
|
|
return Common::kNoError;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SaveLoadManager::readSavegameHeader(int slot, hopkinsSavegameHeader &header) {
|
|
|
|
// Try and open the save file for reading
|
|
|
|
Common::InSaveFile *saveFile = g_system->getSavefileManager()->openForLoading(
|
|
|
|
g_vm->generateSaveName(slot));
|
|
|
|
if (!saveFile)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
bool result = readSavegameHeader(saveFile, header);
|
|
|
|
delete saveFile;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define REDUCE_AMOUNT 80
|
|
|
|
|
|
|
|
void SaveLoadManager::createThumbnail(Graphics::Surface *s) {
|
2012-12-15 22:46:09 +01:00
|
|
|
int w = _vm->_graphicsManager.zoomOut(SCREEN_WIDTH, REDUCE_AMOUNT);
|
|
|
|
int h = _vm->_graphicsManager.zoomOut(SCREEN_HEIGHT - 40, REDUCE_AMOUNT);
|
2012-10-27 22:16:54 +11:00
|
|
|
|
2012-11-30 23:15:47 +11:00
|
|
|
Graphics::Surface thumb8;
|
|
|
|
thumb8.create(w, h, Graphics::PixelFormat::createFormatCLUT8());
|
2012-10-27 22:16:54 +11:00
|
|
|
|
2012-12-19 02:15:32 +01:00
|
|
|
_vm->_graphicsManager.Reduc_Ecran(_vm->_graphicsManager._vesaBuffer, (byte *)thumb8.pixels,
|
2012-12-11 01:53:50 +01:00
|
|
|
_vm->_eventsManager._startPos.x, 20, SCREEN_WIDTH, SCREEN_HEIGHT - 40, 80);
|
2012-11-30 23:15:47 +11:00
|
|
|
|
|
|
|
// Convert the 8-bit pixel to 16 bit surface
|
2012-12-14 01:49:22 +01:00
|
|
|
s->create(w, h, Graphics::PixelFormat(2, 5, 6, 5, 0, 11, 5, 0, 0));
|
2012-11-30 23:15:47 +11:00
|
|
|
|
|
|
|
const byte *srcP = (const byte *)thumb8.pixels;
|
|
|
|
uint16 *destP = (uint16 *)s->pixels;
|
|
|
|
|
|
|
|
for (int yp = 0; yp < h; ++yp) {
|
|
|
|
// Copy over the line, using the source pixels as lookups into the pixels palette
|
|
|
|
const byte *lineSrcP = srcP;
|
|
|
|
uint16 *lineDestP = destP;
|
|
|
|
|
|
|
|
for (int xp = 0; xp < w; ++xp)
|
|
|
|
*lineDestP++ = *(uint16 *)&_vm->_graphicsManager.PAL_PIXELS[*lineSrcP++ * 2];
|
|
|
|
|
|
|
|
// Move to the start of the next line
|
|
|
|
srcP += w;
|
|
|
|
destP += w;
|
|
|
|
}
|
2012-12-08 22:45:11 +11:00
|
|
|
thumb8.free();
|
2012-10-27 22:16:54 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
void SaveLoadManager::syncSavegameData(Common::Serializer &s) {
|
|
|
|
s.syncBytes(&_vm->_globals.SAUVEGARDE->data[0], 0x802);
|
2012-12-17 00:16:49 +01:00
|
|
|
syncCharacterLocation(s, _vm->_globals.SAUVEGARDE->_cloneHopkins);
|
|
|
|
syncCharacterLocation(s, _vm->_globals.SAUVEGARDE->_realHopkins);
|
|
|
|
syncCharacterLocation(s, _vm->_globals.SAUVEGARDE->_samantha);
|
2012-10-27 22:16:54 +11:00
|
|
|
|
|
|
|
for (int i = 0; i < 35; ++i)
|
2012-12-17 00:16:49 +01:00
|
|
|
s.syncAsSint16LE(_vm->_globals.SAUVEGARDE->_inventory[i]);
|
2012-10-27 22:16:54 +11:00
|
|
|
}
|
|
|
|
|
2012-11-24 22:25:19 +11:00
|
|
|
void SaveLoadManager::syncCharacterLocation(Common::Serializer &s, CharacterLocation &item) {
|
2012-12-16 02:11:36 +01:00
|
|
|
s.syncAsSint16LE(item._pos.x);
|
|
|
|
s.syncAsSint16LE(item._pos.y);
|
2012-10-27 22:16:54 +11:00
|
|
|
s.syncAsSint16LE(item.field2);
|
2012-12-16 02:11:36 +01:00
|
|
|
s.syncAsSint16LE(item._location);
|
2012-10-27 22:16:54 +11:00
|
|
|
s.syncAsSint16LE(item.field4);
|
|
|
|
}
|
|
|
|
|
2012-11-30 23:15:47 +11:00
|
|
|
void SaveLoadManager::convertThumb16To8(Graphics::Surface *thumb16, Graphics::Surface *thumb8) {
|
|
|
|
thumb8->create(thumb16->w, thumb16->h, Graphics::PixelFormat::createFormatCLUT8());
|
|
|
|
Graphics::PixelFormat pixelFormat16(2, 5, 6, 5, 0, 11, 5, 0, 0);
|
|
|
|
|
|
|
|
uint16 palette[PALETTE_SIZE];
|
|
|
|
for (int palIndex = 0; palIndex < PALETTE_SIZE; ++palIndex)
|
|
|
|
palette[palIndex] = READ_LE_UINT16(&_vm->_graphicsManager.PAL_PIXELS[palIndex * 2]);
|
|
|
|
|
|
|
|
const uint16 *srcP = (const uint16 *)thumb16->pixels;
|
|
|
|
byte *destP = (byte *)thumb8->pixels;
|
|
|
|
|
|
|
|
for (int yp = 0; yp < thumb16->h; ++yp) {
|
|
|
|
const uint16 *lineSrcP = srcP;
|
|
|
|
byte *lineDestP = destP;
|
|
|
|
|
|
|
|
for (int xp = 0; xp < thumb16->w; ++xp) {
|
|
|
|
byte r, g, b;
|
|
|
|
pixelFormat16.colorToRGB(*lineSrcP++, r, g, b);
|
|
|
|
|
|
|
|
// Scan the palette for the closest match
|
|
|
|
int difference = 99999, foundIndex = 0;
|
|
|
|
for (int palIndex = 0; palIndex < PALETTE_SIZE; ++palIndex) {
|
|
|
|
byte rCurrent, gCurrent, bCurrent;
|
|
|
|
pixelFormat16.colorToRGB(palette[palIndex], rCurrent, gCurrent, bCurrent);
|
2012-12-14 01:49:22 +01:00
|
|
|
|
2012-11-30 23:15:47 +11:00
|
|
|
int diff = ABS((int)r - (int)rCurrent) + ABS((int)g - (int)gCurrent) + ABS((int)b - (int)bCurrent);
|
|
|
|
if (diff < difference) {
|
|
|
|
difference = diff;
|
|
|
|
foundIndex = palIndex;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*lineDestP++ = foundIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Move to the start of the next line
|
|
|
|
srcP += thumb16->w;
|
|
|
|
destP += thumb16->w;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-24 22:49:32 +11:00
|
|
|
} // End of namespace Hopkins
|