2010-08-17 09:28:20 +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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This code is based on original Hugo Trilogy source code
|
|
|
|
*
|
|
|
|
* Copyright (c) 1989-1995 David P. Gray
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2011-04-24 11:34:27 +03:00
|
|
|
#include "common/debug.h"
|
2010-08-17 09:28:20 +00:00
|
|
|
#include "common/system.h"
|
|
|
|
#include "common/savefile.h"
|
2011-04-24 11:34:27 +03:00
|
|
|
#include "common/textconsole.h"
|
2010-11-29 17:42:08 +00:00
|
|
|
#include "common/config-manager.h"
|
2012-09-13 01:51:21 +03:00
|
|
|
|
|
|
|
#include "graphics/surface.h"
|
|
|
|
#include "graphics/decoders/pcx.h"
|
2010-11-29 17:42:08 +00:00
|
|
|
#include "graphics/thumbnail.h"
|
2012-09-13 01:51:21 +03:00
|
|
|
|
2010-11-29 17:42:08 +00:00
|
|
|
#include "gui/saveload.h"
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
#include "hugo/hugo.h"
|
|
|
|
#include "hugo/file.h"
|
|
|
|
#include "hugo/schedule.h"
|
|
|
|
#include "hugo/display.h"
|
|
|
|
#include "hugo/util.h"
|
2010-10-21 17:09:57 +00:00
|
|
|
#include "hugo/object.h"
|
2011-01-25 00:32:48 +00:00
|
|
|
#include "hugo/text.h"
|
2011-02-08 20:52:26 +00:00
|
|
|
#include "hugo/mouse.h"
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
namespace Hugo {
|
2011-03-24 16:46:37 +01:00
|
|
|
|
|
|
|
namespace {
|
2011-03-30 00:21:40 +02:00
|
|
|
static const char s_bootCypher[] = "Copyright 1992, David P Gray, Gray Design Associates";
|
|
|
|
static const int s_bootCypherLen = sizeof(s_bootCypher) - 1;
|
2011-03-24 16:46:37 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-10-21 17:09:57 +00:00
|
|
|
FileManager::FileManager(HugoEngine *vm) : _vm(vm) {
|
2012-06-13 11:50:49 +02:00
|
|
|
_hasReadHeader = false;
|
2012-06-13 17:44:09 +02:00
|
|
|
_firstUIFFl = true;
|
2010-09-05 20:37:38 +00:00
|
|
|
}
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2010-09-05 20:37:38 +00:00
|
|
|
FileManager::~FileManager() {
|
2011-02-18 22:24:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Name scenery and objects picture databases
|
|
|
|
*/
|
2011-02-18 22:43:38 +01:00
|
|
|
const char *FileManager::getBootFilename() const {
|
2011-02-18 22:24:32 +01:00
|
|
|
return "HUGO.BSF";
|
|
|
|
}
|
|
|
|
|
2011-02-18 22:43:38 +01:00
|
|
|
const char *FileManager::getObjectFilename() const {
|
|
|
|
return "objects.dat";
|
2011-02-18 22:24:32 +01:00
|
|
|
}
|
|
|
|
|
2011-02-18 22:43:38 +01:00
|
|
|
const char *FileManager::getSceneryFilename() const {
|
|
|
|
return "scenery.dat";
|
2011-02-18 22:24:32 +01:00
|
|
|
}
|
|
|
|
|
2011-02-18 22:43:38 +01:00
|
|
|
const char *FileManager::getSoundFilename() const {
|
|
|
|
return "sounds.dat";
|
2011-02-18 22:24:32 +01:00
|
|
|
}
|
|
|
|
|
2011-02-18 22:43:38 +01:00
|
|
|
const char *FileManager::getStringFilename() const {
|
|
|
|
return "strings.dat";
|
2011-02-18 22:24:32 +01:00
|
|
|
}
|
|
|
|
|
2011-02-18 22:43:38 +01:00
|
|
|
const char *FileManager::getUifFilename() const {
|
|
|
|
return "uif.dat";
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
|
|
|
|
2010-11-01 20:20:21 +00:00
|
|
|
/**
|
2012-06-13 20:58:01 +02:00
|
|
|
* Read a pcx file of length len. Use supplied seqPtr and image_p or
|
|
|
|
* allocate space if NULL. Name used for errors. Returns address of seqPtr
|
2011-02-11 20:27:48 +00:00
|
|
|
* Set first TRUE to initialize b_index (i.e. not reading a sequential image in file).
|
|
|
|
*/
|
2012-09-13 01:51:21 +03:00
|
|
|
Seq *FileManager::readPCX(Common::SeekableReadStream &f, Seq *seqPtr, byte *imagePtr, const bool firstFl, const char *name) {
|
2010-08-17 09:28:20 +00:00
|
|
|
debugC(1, kDebugFile, "readPCX(..., %s)", name);
|
|
|
|
|
2012-06-13 20:58:01 +02:00
|
|
|
// Allocate memory for Seq if 0
|
2010-10-03 08:08:42 +00:00
|
|
|
if (seqPtr == 0) {
|
2012-06-13 20:58:01 +02:00
|
|
|
if ((seqPtr = (Seq *)malloc(sizeof(Seq))) == 0)
|
2010-11-11 10:36:10 +00:00
|
|
|
error("Insufficient memory to run game.");
|
2010-10-03 08:08:42 +00:00
|
|
|
}
|
2010-10-25 13:31:01 +00:00
|
|
|
|
2012-09-13 01:51:21 +03:00
|
|
|
Graphics::PCXDecoder pcx;
|
|
|
|
if (!pcx.loadStream(f))
|
|
|
|
error("Error while reading PCX image");
|
|
|
|
|
|
|
|
const Graphics::Surface *pcxSurface = pcx.getSurface();
|
|
|
|
if (pcxSurface->format.bytesPerPixel != 1)
|
|
|
|
error("Invalid bytes per pixel in PCX surface (%d)", pcxSurface->format.bytesPerPixel);
|
|
|
|
|
2010-08-17 09:28:20 +00:00
|
|
|
// Find size of image data in 8-bit DIB format
|
|
|
|
// Note save of x2 - marks end of valid data before garbage
|
2012-09-13 01:51:21 +03:00
|
|
|
seqPtr->_lines = pcxSurface->h;
|
|
|
|
seqPtr->_x2 = seqPtr->_bytesPerLine8 = pcxSurface->w;
|
2010-10-03 08:08:42 +00:00
|
|
|
// Size of the image
|
2012-09-13 01:51:21 +03:00
|
|
|
uint16 size = pcxSurface->w * pcxSurface->h;
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// Allocate memory for image data if NULL
|
2011-02-02 21:12:51 +00:00
|
|
|
if (imagePtr == 0)
|
2010-11-11 10:36:10 +00:00
|
|
|
imagePtr = (byte *)malloc((size_t) size);
|
2011-02-02 21:12:51 +00:00
|
|
|
|
2010-11-11 10:36:10 +00:00
|
|
|
assert(imagePtr);
|
2010-10-03 08:08:42 +00:00
|
|
|
|
2012-06-13 11:28:25 +02:00
|
|
|
seqPtr->_imagePtr = imagePtr;
|
2012-09-13 01:51:21 +03:00
|
|
|
for (uint16 y = 0; y < pcxSurface->h; y++)
|
2012-09-14 03:35:18 +03:00
|
|
|
memcpy(imagePtr + y * pcxSurface->w, pcxSurface->getBasePtr(0, y), pcxSurface->w);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
return seqPtr;
|
|
|
|
}
|
|
|
|
|
2010-11-01 20:20:21 +00:00
|
|
|
/**
|
2011-02-11 20:27:48 +00:00
|
|
|
* Read object file of PCC images into object supplied
|
|
|
|
*/
|
2012-06-13 20:58:01 +02:00
|
|
|
void FileManager::readImage(const int objNum, Object *objPtr) {
|
|
|
|
debugC(1, kDebugFile, "readImage(%d, Object *objPtr)", objNum);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2010-11-22 19:50:57 +00:00
|
|
|
/**
|
2011-02-11 20:27:48 +00:00
|
|
|
* Structure of object file lookup entry
|
|
|
|
*/
|
2010-11-22 19:50:57 +00:00
|
|
|
struct objBlock_t {
|
|
|
|
uint32 objOffset;
|
|
|
|
uint32 objLength;
|
|
|
|
};
|
|
|
|
|
2012-06-13 11:28:25 +02:00
|
|
|
if (!objPtr->_seqNumb) // This object has no images
|
2010-08-17 09:28:20 +00:00
|
|
|
return;
|
|
|
|
|
2010-10-21 17:09:57 +00:00
|
|
|
if (_vm->isPacked()) {
|
2010-08-17 09:28:20 +00:00
|
|
|
_objectsArchive.seek((uint32)objNum * sizeof(objBlock_t), SEEK_SET);
|
|
|
|
|
2010-10-03 08:08:42 +00:00
|
|
|
objBlock_t objBlock; // Info on file within database
|
2010-08-17 09:28:20 +00:00
|
|
|
objBlock.objOffset = _objectsArchive.readUint32LE();
|
|
|
|
objBlock.objLength = _objectsArchive.readUint32LE();
|
|
|
|
|
|
|
|
_objectsArchive.seek(objBlock.objOffset, SEEK_SET);
|
|
|
|
} else {
|
2011-01-30 23:08:05 +00:00
|
|
|
Common::String buf;
|
2012-06-13 11:28:25 +02:00
|
|
|
buf = _vm->_picDir + Common::String(_vm->_text->getNoun(objPtr->_nounIndex, 0)) + ".PIX";
|
2010-08-17 09:28:20 +00:00
|
|
|
if (!_objectsArchive.open(buf)) {
|
2012-06-13 11:28:25 +02:00
|
|
|
buf = Common::String(_vm->_text->getNoun(objPtr->_nounIndex, 0)) + ".PIX";
|
2010-08-17 09:28:20 +00:00
|
|
|
if (!_objectsArchive.open(buf))
|
2011-01-30 23:19:30 +00:00
|
|
|
error("File not found: %s", buf.c_str());
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-02-03 18:25:38 +00:00
|
|
|
bool firstImgFl = true; // Initializes pcx read function
|
2012-06-13 20:58:01 +02:00
|
|
|
Seq *seqPtr = 0; // Ptr to sequence structure
|
2010-10-03 08:08:42 +00:00
|
|
|
|
2010-08-17 09:28:20 +00:00
|
|
|
// Now read the images into an images list
|
2012-06-13 11:28:25 +02:00
|
|
|
for (int j = 0; j < objPtr->_seqNumb; j++) { // for each sequence
|
|
|
|
for (int k = 0; k < objPtr->_seqList[j]._imageNbr; k++) { // each image
|
2010-08-17 09:28:20 +00:00
|
|
|
if (k == 0) { // First image
|
|
|
|
// Read this image - allocate both seq and image memory
|
2012-06-13 11:28:25 +02:00
|
|
|
seqPtr = readPCX(_objectsArchive, 0, 0, firstImgFl, _vm->_text->getNoun(objPtr->_nounIndex, 0));
|
|
|
|
objPtr->_seqList[j]._seqPtr = seqPtr;
|
2011-02-03 18:25:38 +00:00
|
|
|
firstImgFl = false;
|
2010-08-17 09:28:20 +00:00
|
|
|
} else { // Subsequent image
|
|
|
|
// Read this image - allocate both seq and image memory
|
2012-06-13 11:28:25 +02:00
|
|
|
seqPtr->_nextSeqPtr = readPCX(_objectsArchive, 0, 0, firstImgFl, _vm->_text->getNoun(objPtr->_nounIndex, 0));
|
|
|
|
seqPtr = seqPtr->_nextSeqPtr;
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Compute the bounding box - x1, x2, y1, y2
|
|
|
|
// Note use of x2 - marks end of valid data in row
|
2012-06-13 11:28:25 +02:00
|
|
|
uint16 x2 = seqPtr->_x2;
|
|
|
|
seqPtr->_x1 = seqPtr->_x2;
|
|
|
|
seqPtr->_x2 = 0;
|
|
|
|
seqPtr->_y1 = seqPtr->_lines;
|
|
|
|
seqPtr->_y2 = 0;
|
|
|
|
|
2012-06-13 20:58:01 +02:00
|
|
|
ImagePtr dibPtr = seqPtr->_imagePtr;
|
2012-06-13 11:28:25 +02:00
|
|
|
for (int y = 0; y < seqPtr->_lines; y++, dibPtr += seqPtr->_bytesPerLine8 - x2) {
|
2010-10-03 08:08:42 +00:00
|
|
|
for (int x = 0; x < x2; x++) {
|
|
|
|
if (*dibPtr++) { // Some data found
|
2012-06-13 11:28:25 +02:00
|
|
|
if (x < seqPtr->_x1)
|
|
|
|
seqPtr->_x1 = x;
|
|
|
|
if (x > seqPtr->_x2)
|
|
|
|
seqPtr->_x2 = x;
|
|
|
|
if (y < seqPtr->_y1)
|
|
|
|
seqPtr->_y1 = y;
|
|
|
|
if (y > seqPtr->_y2)
|
|
|
|
seqPtr->_y2 = y;
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
2010-10-03 08:08:42 +00:00
|
|
|
}
|
|
|
|
}
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
2010-10-15 06:08:42 +00:00
|
|
|
assert(seqPtr);
|
2012-06-13 11:28:25 +02:00
|
|
|
seqPtr->_nextSeqPtr = objPtr->_seqList[j]._seqPtr; // loop linked list to head
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Set the current image sequence to first or last
|
2012-06-13 11:28:25 +02:00
|
|
|
switch (objPtr->_cycling) {
|
2011-01-23 00:05:52 +00:00
|
|
|
case kCycleInvisible: // (May become visible later)
|
|
|
|
case kCycleAlmostInvisible:
|
|
|
|
case kCycleNotCycling:
|
|
|
|
case kCycleForward:
|
2012-06-13 11:28:25 +02:00
|
|
|
objPtr->_currImagePtr = objPtr->_seqList[0]._seqPtr;
|
2010-08-17 09:28:20 +00:00
|
|
|
break;
|
2011-01-23 00:05:52 +00:00
|
|
|
case kCycleBackward:
|
2012-06-13 11:28:25 +02:00
|
|
|
objPtr->_currImagePtr = seqPtr;
|
2010-08-17 09:28:20 +00:00
|
|
|
break;
|
2010-09-16 23:03:03 +00:00
|
|
|
default:
|
2012-06-13 11:28:25 +02:00
|
|
|
warning("Unexpected cycling: %d", objPtr->_cycling);
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
|
|
|
|
2010-10-21 17:09:57 +00:00
|
|
|
if (!_vm->isPacked())
|
2010-08-17 09:28:20 +00:00
|
|
|
_objectsArchive.close();
|
|
|
|
}
|
|
|
|
|
2010-11-01 20:20:21 +00:00
|
|
|
/**
|
2011-02-11 20:27:48 +00:00
|
|
|
* Read sound (or music) file data. Call with SILENCE to free-up
|
|
|
|
* any allocated memory. Also returns size of data
|
|
|
|
*/
|
2012-06-13 21:18:37 +02:00
|
|
|
SoundPtr FileManager::getSound(const int16 sound, uint16 *size) {
|
2010-12-05 21:52:42 +00:00
|
|
|
debugC(1, kDebugFile, "getSound(%d)", sound);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// No more to do if SILENCE (called for cleanup purposes)
|
2010-10-21 17:09:57 +00:00
|
|
|
if (sound == _vm->_soundSilence)
|
2010-10-03 08:08:42 +00:00
|
|
|
return 0;
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// Open sounds file
|
2010-10-03 08:08:42 +00:00
|
|
|
Common::File fp; // Handle to SOUND_FILE
|
2011-01-23 22:51:12 +00:00
|
|
|
if (!fp.open(getSoundFilename())) {
|
|
|
|
warning("Hugo Error: File not found %s", getSoundFilename());
|
2010-10-03 08:08:42 +00:00
|
|
|
return 0;
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
|
|
|
|
2012-06-13 11:50:49 +02:00
|
|
|
if (!_hasReadHeader) {
|
2011-04-24 23:22:39 +02:00
|
|
|
for (int i = 0; i < kMaxSounds; i++) {
|
2012-06-13 21:18:37 +02:00
|
|
|
_soundHdr[i]._size = fp.readUint16LE();
|
|
|
|
_soundHdr[i]._offset = fp.readUint32LE();
|
2011-04-24 23:22:39 +02:00
|
|
|
}
|
|
|
|
if (fp.err())
|
2011-01-23 22:51:12 +00:00
|
|
|
error("Wrong sound file format");
|
2012-06-13 11:50:49 +02:00
|
|
|
_hasReadHeader = true;
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
|
|
|
|
2012-06-13 21:18:37 +02:00
|
|
|
*size = _soundHdr[sound]._size;
|
2010-08-17 09:28:20 +00:00
|
|
|
if (*size == 0)
|
2011-01-23 22:51:12 +00:00
|
|
|
error("Wrong sound file format or missing sound %d", sound);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// Allocate memory for sound or music, if possible
|
2012-06-13 21:18:37 +02:00
|
|
|
SoundPtr soundPtr = (byte *)malloc(_soundHdr[sound]._size); // Ptr to sound data
|
2010-11-11 10:36:10 +00:00
|
|
|
assert(soundPtr);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// Seek to data and read it
|
2012-06-13 21:18:37 +02:00
|
|
|
fp.seek(_soundHdr[sound]._offset, SEEK_SET);
|
|
|
|
if (fp.read(soundPtr, _soundHdr[sound]._size) != _soundHdr[sound]._size)
|
2011-01-23 22:51:12 +00:00
|
|
|
error("Wrong sound file format");
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
fp.close();
|
|
|
|
|
|
|
|
return soundPtr;
|
|
|
|
}
|
|
|
|
|
2010-11-01 20:20:21 +00:00
|
|
|
/**
|
2011-02-11 20:27:48 +00:00
|
|
|
* Save game to supplied slot
|
|
|
|
*/
|
2011-02-12 08:13:35 +00:00
|
|
|
bool FileManager::saveGame(const int16 slot, const Common::String &descrip) {
|
2010-11-29 18:03:05 +00:00
|
|
|
debugC(1, kDebugFile, "saveGame(%d, %s)", slot, descrip.c_str());
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2010-11-29 17:42:08 +00:00
|
|
|
int16 savegameId;
|
|
|
|
Common::String savegameDescription;
|
|
|
|
|
|
|
|
if (slot == -1) {
|
2012-06-10 04:14:17 +02:00
|
|
|
GUI::SaveLoadChooser *dialog = new GUI::SaveLoadChooser("Save game:", "Save", true);
|
2012-06-10 04:49:42 +02:00
|
|
|
savegameId = dialog->runModalWithCurrentTarget();
|
2010-11-29 17:42:08 +00:00
|
|
|
savegameDescription = dialog->getResultString();
|
|
|
|
delete dialog;
|
|
|
|
} else {
|
|
|
|
savegameId = slot;
|
|
|
|
if (!descrip.empty()) {
|
|
|
|
savegameDescription = descrip;
|
|
|
|
} else {
|
|
|
|
savegameDescription = Common::String::format("Quick save #%d", slot);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (savegameId < 0) // dialog aborted
|
|
|
|
return false;
|
|
|
|
|
2011-02-12 08:13:35 +00:00
|
|
|
Common::String savegameFile = _vm->getSavegameFilename(savegameId);
|
2010-11-29 17:42:08 +00:00
|
|
|
Common::SaveFileManager *saveMan = g_system->getSavefileManager();
|
|
|
|
Common::OutSaveFile *out = saveMan->openForSaving(savegameFile);
|
|
|
|
|
2010-10-10 23:24:57 +00:00
|
|
|
if (!out) {
|
2010-11-29 17:42:08 +00:00
|
|
|
warning("Can't create file '%s', game not saved", savegameFile.c_str());
|
|
|
|
return false;
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Write version. We can't restore from obsolete versions
|
2010-11-19 23:49:04 +00:00
|
|
|
out->writeByte(kSavegameVersion);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2010-11-29 17:42:08 +00:00
|
|
|
if (savegameDescription == "") {
|
|
|
|
savegameDescription = "Untitled savegame";
|
|
|
|
}
|
|
|
|
|
|
|
|
out->writeSint16BE(savegameDescription.size() + 1);
|
|
|
|
out->write(savegameDescription.c_str(), savegameDescription.size() + 1);
|
|
|
|
|
|
|
|
Graphics::saveThumbnail(*out);
|
|
|
|
|
|
|
|
TimeDate curTime;
|
|
|
|
_vm->_system->getTimeAndDate(curTime);
|
|
|
|
|
|
|
|
uint32 saveDate = (curTime.tm_mday & 0xFF) << 24 | ((curTime.tm_mon + 1) & 0xFF) << 16 | ((curTime.tm_year + 1900) & 0xFFFF);
|
|
|
|
uint16 saveTime = (curTime.tm_hour & 0xFF) << 8 | ((curTime.tm_min) & 0xFF);
|
|
|
|
|
|
|
|
out->writeUint32BE(saveDate);
|
|
|
|
out->writeUint16BE(saveTime);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2010-11-11 23:04:04 +00:00
|
|
|
_vm->_object->saveObjects(out);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2012-06-13 20:58:01 +02:00
|
|
|
const Status &gameStatus = _vm->getGameStatus();
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// Save whether hero image is swapped
|
2010-11-19 23:49:04 +00:00
|
|
|
out->writeByte(_vm->_heroImage);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// Save score
|
2010-11-19 23:49:04 +00:00
|
|
|
out->writeSint16BE(_vm->getScore());
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// Save story mode
|
2012-06-13 11:28:25 +02:00
|
|
|
out->writeByte((gameStatus._storyModeFl) ? 1 : 0);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// Save jumpexit mode
|
2011-02-08 20:52:26 +00:00
|
|
|
out->writeByte((_vm->_mouse->getJumpExitFl()) ? 1 : 0);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// Save gameover status
|
2012-06-13 11:28:25 +02:00
|
|
|
out->writeByte((gameStatus._gameOverFl) ? 1 : 0);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// Save screen states
|
2011-04-24 22:29:27 +02:00
|
|
|
for (int i = 0; i < _vm->_numStates; i++)
|
2010-11-19 23:49:04 +00:00
|
|
|
out->writeByte(_vm->_screenStates[i]);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2011-02-15 19:30:15 +01:00
|
|
|
_vm->_scheduler->saveSchedulerData(out);
|
2010-08-17 09:28:20 +00:00
|
|
|
// Save palette table
|
2010-10-21 17:09:57 +00:00
|
|
|
_vm->_screen->savePal(out);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// Save maze status
|
2012-06-13 11:28:25 +02:00
|
|
|
out->writeByte((_vm->_maze._enabledFl) ? 1 : 0);
|
|
|
|
out->writeByte(_vm->_maze._size);
|
|
|
|
out->writeSint16BE(_vm->_maze._x1);
|
|
|
|
out->writeSint16BE(_vm->_maze._y1);
|
|
|
|
out->writeSint16BE(_vm->_maze._x2);
|
|
|
|
out->writeSint16BE(_vm->_maze._y2);
|
|
|
|
out->writeSint16BE(_vm->_maze._x3);
|
|
|
|
out->writeSint16BE(_vm->_maze._x4);
|
|
|
|
out->writeByte(_vm->_maze._firstScreenIndex);
|
|
|
|
|
|
|
|
out->writeByte((byte)_vm->getGameStatus()._viewState);
|
2011-03-08 00:19:30 +01:00
|
|
|
|
2010-08-17 09:28:20 +00:00
|
|
|
out->finalize();
|
|
|
|
|
|
|
|
delete out;
|
2010-11-29 17:42:08 +00:00
|
|
|
|
|
|
|
return true;
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
|
|
|
|
2010-11-01 20:20:21 +00:00
|
|
|
/**
|
2011-02-11 20:27:48 +00:00
|
|
|
* Restore game from supplied slot number
|
|
|
|
*/
|
2011-02-02 21:12:51 +00:00
|
|
|
bool FileManager::restoreGame(const int16 slot) {
|
2010-08-17 09:28:20 +00:00
|
|
|
debugC(1, kDebugFile, "restoreGame(%d)", slot);
|
|
|
|
|
2010-11-29 17:42:08 +00:00
|
|
|
int16 savegameId;
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2010-11-29 17:42:08 +00:00
|
|
|
if (slot == -1) {
|
2012-06-10 04:14:17 +02:00
|
|
|
GUI::SaveLoadChooser *dialog = new GUI::SaveLoadChooser("Restore game:", "Restore", false);
|
2012-06-10 04:49:42 +02:00
|
|
|
savegameId = dialog->runModalWithCurrentTarget();
|
2010-11-29 17:42:08 +00:00
|
|
|
delete dialog;
|
|
|
|
} else {
|
|
|
|
savegameId = slot;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (savegameId < 0) // dialog aborted
|
|
|
|
return false;
|
2010-10-10 23:24:57 +00:00
|
|
|
|
2011-02-12 08:13:35 +00:00
|
|
|
Common::String savegameFile = _vm->getSavegameFilename(savegameId);
|
2010-11-29 17:42:08 +00:00
|
|
|
Common::SaveFileManager *saveMan = g_system->getSavefileManager();
|
|
|
|
Common::InSaveFile *in = saveMan->openForLoading(savegameFile);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2010-10-10 23:24:57 +00:00
|
|
|
if (!in)
|
2010-11-29 17:42:08 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// Initialize new-game status
|
|
|
|
_vm->initStatus();
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// Check version, can't restore from different versions
|
2010-11-19 23:49:04 +00:00
|
|
|
int saveVersion = in->readByte();
|
2010-08-17 09:28:20 +00:00
|
|
|
if (saveVersion != kSavegameVersion) {
|
2010-11-29 17:42:08 +00:00
|
|
|
warning("Savegame of incompatible version");
|
|
|
|
delete in;
|
|
|
|
return false;
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Skip over description
|
2010-11-29 17:42:08 +00:00
|
|
|
int32 saveGameNameSize = in->readSint16BE();
|
|
|
|
in->skip(saveGameNameSize);
|
|
|
|
|
|
|
|
Graphics::skipThumbnail(*in);
|
|
|
|
|
|
|
|
in->skip(6); // Skip date & time
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// If hero image is currently swapped, swap it back before restore
|
2011-01-23 00:05:52 +00:00
|
|
|
if (_vm->_heroImage != kHeroIndex)
|
|
|
|
_vm->_object->swapImages(kHeroIndex, _vm->_heroImage);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2010-11-11 23:04:04 +00:00
|
|
|
_vm->_object->restoreObjects(in);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2010-11-19 23:49:04 +00:00
|
|
|
_vm->_heroImage = in->readByte();
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// If hero swapped in saved game, swap it
|
2010-11-19 23:49:04 +00:00
|
|
|
byte heroImg = _vm->_heroImage;
|
2011-01-23 00:05:52 +00:00
|
|
|
if (heroImg != kHeroIndex)
|
|
|
|
_vm->_object->swapImages(kHeroIndex, _vm->_heroImage);
|
2010-10-21 17:09:57 +00:00
|
|
|
_vm->_heroImage = heroImg;
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2012-06-13 20:58:01 +02:00
|
|
|
Status &gameStatus = _vm->getGameStatus();
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2010-11-22 22:49:24 +00:00
|
|
|
int score = in->readSint16BE();
|
2010-10-21 17:09:57 +00:00
|
|
|
_vm->setScore(score);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2012-06-13 11:28:25 +02:00
|
|
|
gameStatus._storyModeFl = (in->readByte() == 1);
|
2011-02-08 20:52:26 +00:00
|
|
|
_vm->_mouse->setJumpExitFl(in->readByte() == 1);
|
2012-06-13 11:28:25 +02:00
|
|
|
gameStatus._gameOverFl = (in->readByte() == 1);
|
2011-04-24 22:29:27 +02:00
|
|
|
for (int i = 0; i < _vm->_numStates; i++)
|
2010-11-19 23:49:04 +00:00
|
|
|
_vm->_screenStates[i] = in->readByte();
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2011-02-15 19:30:15 +01:00
|
|
|
_vm->_scheduler->restoreSchedulerData(in);
|
2011-01-26 00:21:54 +00:00
|
|
|
|
2010-08-17 09:28:20 +00:00
|
|
|
// Restore palette and change it if necessary
|
2010-10-21 17:09:57 +00:00
|
|
|
_vm->_screen->restorePal(in);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// Restore maze status
|
2012-06-13 11:28:25 +02:00
|
|
|
_vm->_maze._enabledFl = (in->readByte() == 1);
|
|
|
|
_vm->_maze._size = in->readByte();
|
|
|
|
_vm->_maze._x1 = in->readSint16BE();
|
|
|
|
_vm->_maze._y1 = in->readSint16BE();
|
|
|
|
_vm->_maze._x2 = in->readSint16BE();
|
|
|
|
_vm->_maze._y2 = in->readSint16BE();
|
|
|
|
_vm->_maze._x3 = in->readSint16BE();
|
|
|
|
_vm->_maze._x4 = in->readSint16BE();
|
|
|
|
_vm->_maze._firstScreenIndex = in->readByte();
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2012-06-13 21:18:37 +02:00
|
|
|
_vm->_scheduler->restoreScreen(*_vm->_screenPtr);
|
2012-06-13 20:58:01 +02:00
|
|
|
if ((_vm->getGameStatus()._viewState = (Vstate) in->readByte()) != kViewPlay)
|
2011-03-08 00:19:30 +01:00
|
|
|
_vm->_screen->hideCursor();
|
|
|
|
|
|
|
|
|
2010-08-17 09:28:20 +00:00
|
|
|
delete in;
|
2010-11-29 17:42:08 +00:00
|
|
|
return true;
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
|
|
|
|
2010-11-01 20:20:21 +00:00
|
|
|
/**
|
2011-02-11 20:27:48 +00:00
|
|
|
* Reads boot file for program environment. Fatal error if not there or
|
|
|
|
* file checksum is bad. De-crypts structure while checking checksum
|
|
|
|
*/
|
2010-08-17 09:28:20 +00:00
|
|
|
void FileManager::readBootFile() {
|
2011-01-23 22:51:12 +00:00
|
|
|
debugC(1, kDebugFile, "readBootFile()");
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2010-10-03 08:08:42 +00:00
|
|
|
Common::File ofp;
|
2011-01-23 22:51:12 +00:00
|
|
|
if (!ofp.open(getBootFilename())) {
|
2011-01-18 18:26:33 +00:00
|
|
|
if (_vm->_gameVariant == kGameVariantH1Dos) {
|
2010-08-18 15:33:59 +00:00
|
|
|
//TODO initialize properly _boot structure
|
|
|
|
warning("readBootFile - Skipping as H1 Dos may be a freeware");
|
2012-06-13 11:28:25 +02:00
|
|
|
memset(_vm->_boot._distrib, '\0', sizeof(_vm->_boot._distrib));
|
|
|
|
_vm->_boot._registered = kRegFreeware;
|
2010-08-18 15:33:59 +00:00
|
|
|
return;
|
2011-03-29 23:54:47 +02:00
|
|
|
} else if (_vm->getPlatform() == Common::kPlatformPC) {
|
|
|
|
warning("readBootFile - Skipping as H2 and H3 Dos may be shareware");
|
2012-06-13 11:28:25 +02:00
|
|
|
memset(_vm->_boot._distrib, '\0', sizeof(_vm->_boot._distrib));
|
|
|
|
_vm->_boot._registered = kRegShareware;
|
2011-03-29 23:54:47 +02:00
|
|
|
return;
|
2010-10-03 08:08:42 +00:00
|
|
|
} else {
|
2011-03-29 23:54:47 +02:00
|
|
|
Utils::notifyBox(Common::String::format("Missing startup file '%s'", getBootFilename()));
|
2012-06-13 11:28:25 +02:00
|
|
|
_vm->getGameStatus()._doQuitFl = true;
|
2011-03-29 23:54:47 +02:00
|
|
|
return;
|
2010-10-03 08:08:42 +00:00
|
|
|
}
|
2010-08-18 17:51:44 +00:00
|
|
|
}
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2011-03-29 23:54:47 +02:00
|
|
|
if (ofp.size() < (int32)sizeof(_vm->_boot)) {
|
|
|
|
Utils::notifyBox(Common::String::format("Corrupted startup file '%s'", getBootFilename()));
|
2012-06-13 11:28:25 +02:00
|
|
|
_vm->getGameStatus()._doQuitFl = true;
|
2011-03-29 23:54:47 +02:00
|
|
|
return;
|
|
|
|
}
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2012-06-13 11:28:25 +02:00
|
|
|
_vm->_boot._checksum = ofp.readByte();
|
|
|
|
_vm->_boot._registered = ofp.readByte();
|
|
|
|
ofp.read(_vm->_boot._pbswitch, sizeof(_vm->_boot._pbswitch));
|
|
|
|
ofp.read(_vm->_boot._distrib, sizeof(_vm->_boot._distrib));
|
|
|
|
_vm->_boot._exitLen = ofp.readUint16LE();
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2011-02-20 10:37:41 +01:00
|
|
|
byte *p = (byte *)&_vm->_boot;
|
2010-10-03 08:08:42 +00:00
|
|
|
|
|
|
|
byte checksum = 0;
|
2011-02-20 10:37:41 +01:00
|
|
|
for (uint32 i = 0; i < sizeof(_vm->_boot); i++) {
|
2010-08-17 09:28:20 +00:00
|
|
|
checksum ^= p[i];
|
2011-03-29 23:54:47 +02:00
|
|
|
p[i] ^= s_bootCypher[i % s_bootCypherLen];
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
|
|
|
ofp.close();
|
|
|
|
|
2011-03-29 23:54:47 +02:00
|
|
|
if (checksum) {
|
|
|
|
Utils::notifyBox(Common::String::format("Corrupted startup file '%s'", getBootFilename()));
|
2012-06-13 11:28:25 +02:00
|
|
|
_vm->getGameStatus()._doQuitFl = true;
|
2011-03-29 23:54:47 +02:00
|
|
|
}
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
|
|
|
|
2010-11-01 20:20:21 +00:00
|
|
|
/**
|
2011-02-11 20:27:48 +00:00
|
|
|
* Returns address of uif_hdr[id], reading it in if first call
|
|
|
|
* This file contains, between others, the bitmaps of the fonts used in the application
|
|
|
|
* UIF means User interface database (Windows Only)
|
|
|
|
*/
|
2012-06-13 20:58:01 +02:00
|
|
|
UifHdr *FileManager::getUIFHeader(const Uif id) {
|
|
|
|
debugC(1, kDebugFile, "getUIFHeader(%d)", id);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// Initialize offset lookup if not read yet
|
2012-06-13 17:44:09 +02:00
|
|
|
if (_firstUIFFl) {
|
|
|
|
_firstUIFFl = false;
|
2010-08-17 09:28:20 +00:00
|
|
|
// Open unbuffered to do far read
|
2010-10-03 08:08:42 +00:00
|
|
|
Common::File ip; // Image data file
|
2011-01-23 22:51:12 +00:00
|
|
|
if (!ip.open(getUifFilename()))
|
|
|
|
error("File not found: %s", getUifFilename());
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2012-06-13 17:44:09 +02:00
|
|
|
if (ip.size() < (int32)sizeof(_UIFHeader))
|
2011-01-23 22:51:12 +00:00
|
|
|
error("Wrong UIF file format");
|
2010-08-17 09:28:20 +00:00
|
|
|
|
2011-01-23 00:05:52 +00:00
|
|
|
for (int i = 0; i < kMaxUifs; ++i) {
|
2012-06-13 17:44:09 +02:00
|
|
|
_UIFHeader[i]._size = ip.readUint16LE();
|
|
|
|
_UIFHeader[i]._offset = ip.readUint32LE();
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ip.close();
|
|
|
|
}
|
2012-06-13 17:44:09 +02:00
|
|
|
return &_UIFHeader[id];
|
2010-08-17 09:28:20 +00:00
|
|
|
}
|
|
|
|
|
2010-11-01 20:20:21 +00:00
|
|
|
/**
|
2011-02-11 20:27:48 +00:00
|
|
|
* Read uif item into supplied buffer.
|
|
|
|
*/
|
2011-02-02 21:12:51 +00:00
|
|
|
void FileManager::readUIFItem(const int16 id, byte *buf) {
|
2010-08-17 09:28:20 +00:00
|
|
|
debugC(1, kDebugFile, "readUIFItem(%d, ...)", id);
|
|
|
|
|
|
|
|
// Open uif file to read data
|
2010-10-03 08:08:42 +00:00
|
|
|
Common::File ip; // UIF_FILE handle
|
2011-01-23 22:51:12 +00:00
|
|
|
if (!ip.open(getUifFilename()))
|
|
|
|
error("File not found: %s", getUifFilename());
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// Seek to data
|
2012-06-13 20:58:01 +02:00
|
|
|
UifHdr *_UIFHeaderPtr = getUIFHeader((Uif)id);
|
2012-06-13 17:44:09 +02:00
|
|
|
ip.seek(_UIFHeaderPtr->_offset, SEEK_SET);
|
2010-08-17 09:28:20 +00:00
|
|
|
|
|
|
|
// We support pcx images and straight data
|
2012-06-13 20:58:01 +02:00
|
|
|
Seq *dummySeq; // Dummy Seq for image data
|
2010-08-17 09:28:20 +00:00
|
|
|
switch (id) {
|
|
|
|
case UIF_IMAGES: // Read uif images file
|
2011-01-23 22:51:12 +00:00
|
|
|
dummySeq = readPCX(ip, 0, buf, true, getUifFilename());
|
2010-11-09 20:26:12 +00:00
|
|
|
free(dummySeq);
|
2010-08-17 09:28:20 +00:00
|
|
|
break;
|
|
|
|
default: // Read file data into supplied array
|
2012-06-13 17:44:09 +02:00
|
|
|
if (ip.read(buf, _UIFHeaderPtr->_size) != _UIFHeaderPtr->_size)
|
2011-01-23 22:51:12 +00:00
|
|
|
error("Wrong UIF file format");
|
2010-08-17 09:28:20 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ip.close();
|
|
|
|
}
|
|
|
|
|
2010-11-01 20:20:21 +00:00
|
|
|
/**
|
2011-02-11 20:27:48 +00:00
|
|
|
* Read the uif image file (inventory icons)
|
|
|
|
*/
|
2010-10-21 17:09:57 +00:00
|
|
|
void FileManager::readUIFImages() {
|
2011-01-23 22:51:12 +00:00
|
|
|
debugC(1, kDebugFile, "readUIFImages()");
|
2010-10-21 17:09:57 +00:00
|
|
|
|
|
|
|
readUIFItem(UIF_IMAGES, _vm->_screen->getGUIBuffer()); // Read all uif images
|
|
|
|
}
|
|
|
|
|
2010-08-27 09:48:53 +00:00
|
|
|
} // End of namespace Hugo
|