2007-05-30 21:56:52 +00:00
/* ScummVM - Graphic Adventure Engine
2006-05-23 23:43:52 +00:00
*
2007-05-30 21:56:52 +00:00
* 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 .
2006-05-23 23:43:52 +00:00
*
* 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 .
*
*/
2009-06-06 17:39:13 +00:00
//
// Savegame support by Vasyl Tsvirkunov <vasyl@pacbell.net>
// Multi-slots by Claudio Matsuoka <claudio@helllabs.org>
//
2006-05-23 23:43:52 +00:00
2006-12-06 19:27:02 +00:00
# include "common/file.h"
2009-07-22 15:55:33 +00:00
# include "common/config-manager.h"
2010-01-03 20:15:44 +00:00
# include "common/savefile.h"
2011-04-24 11:34:27 +03:00
# include "common/textconsole.h"
2012-03-27 00:21:16 +03:00
# include "common/translation.h"
# include "gui/saveload.h"
2010-11-19 01:37:04 +00:00
# include "graphics/thumbnail.h"
# include "graphics/surface.h"
2006-05-23 23:43:52 +00:00
# include "agi/agi.h"
# include "agi/graphics.h"
# include "agi/sprite.h"
# include "agi/keyboard.h"
# include "agi/menu.h"
2012-03-27 00:50:29 +03:00
# define SAVEGAME_VERSION 6
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
//
// 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
2009-06-06 17:41:50 +00:00
// Version 5 (ScummVM): Added game md5
2012-03-27 00:50:29 +03:00
// Version 6 (ScummVM): Added game played time
2009-06-06 17:39:13 +00:00
//
2006-05-23 23:43:52 +00:00
2007-01-12 02:29:20 +00:00
namespace Agi {
2011-04-12 16:53:15 +02:00
static const uint32 AGIflag = MKTAG ( ' A ' , ' G ' , ' I ' , ' : ' ) ;
2006-05-23 23:43:52 +00:00
2011-09-26 09:27:15 +02:00
int AgiEngine : : saveGame ( const Common : : String & fileName , const Common : : String & description ) {
2009-06-06 17:39:13 +00:00
char gameIDstring [ 8 ] = " gameIDX " ;
2007-01-12 02:29:20 +00:00
int i ;
Common : : OutSaveFile * out ;
2007-08-04 06:18:28 +00:00
int result = errOK ;
2006-05-23 23:43:52 +00:00
2011-09-25 22:23:22 +03:00
debugC ( 3 , kDebugLevelMain | kDebugLevelSavegame , " AgiEngine::saveGame(%s, %s) " , fileName . c_str ( ) , description . c_str ( ) ) ;
2007-01-12 02:29:20 +00:00
if ( ! ( out = _saveFileMan - > openForSaving ( fileName ) ) ) {
2011-09-25 21:25:25 +03:00
warning ( " Can't create file '%s', game not saved " , fileName . c_str ( ) ) ;
2007-01-16 12:40:51 +00:00
return errBadFileOpen ;
2007-01-12 02:29:20 +00:00
} else {
2011-09-25 21:25:25 +03:00
debugC ( 3 , kDebugLevelMain | kDebugLevelSavegame , " Successfully opened %s for writing " , fileName . c_str ( ) ) ;
2007-01-12 02:29:20 +00:00
}
out - > writeUint32BE ( AGIflag ) ;
2011-09-25 21:25:25 +03:00
out - > write ( description . c_str ( ) , 31 ) ;
2006-05-23 23:43:52 +00:00
2007-01-12 02:29:20 +00:00
out - > writeByte ( SAVEGAME_VERSION ) ;
debugC ( 5 , kDebugLevelMain | kDebugLevelSavegame , " Writing save game version (%d) " , SAVEGAME_VERSION ) ;
2006-05-23 23:43:52 +00:00
2008-11-10 19:02:47 +00:00
// Thumbnail
Graphics : : saveThumbnail ( * out ) ;
// Creation date/time
2009-10-08 19:41:38 +00:00
TimeDate curTime ;
2008-11-10 19:02:47 +00:00
_system - > getTimeAndDate ( curTime ) ;
2008-11-29 18:01:16 +00:00
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 ) ;
2012-03-27 00:50:29 +03:00
uint32 playTime = g_engine - > getTotalPlayTime ( ) / 1000 ;
2008-11-10 19:02:47 +00:00
out - > writeUint32BE ( saveDate ) ;
debugC ( 5 , kDebugLevelMain | kDebugLevelSavegame , " Writing save date (%d) " , saveDate ) ;
out - > writeUint16BE ( saveTime ) ;
debugC ( 5 , kDebugLevelMain | kDebugLevelSavegame , " Writing save time (%d) " , saveTime ) ;
2012-03-27 00:50:29 +03:00
out - > writeUint32BE ( playTime ) ;
debugC ( 5 , kDebugLevelMain | kDebugLevelSavegame , " Writing play time (%d) " , playTime ) ;
2008-11-10 19:02:47 +00:00
2007-01-16 12:40:51 +00:00
out - > writeByte ( _game . state ) ;
debugC ( 5 , kDebugLevelMain | kDebugLevelSavegame , " Writing game state (%d) " , _game . state ) ;
2007-09-19 08:40:12 +00:00
2007-01-16 12:40:51 +00:00
strcpy ( gameIDstring , _game . id ) ;
2007-01-12 02:29:20 +00:00
out - > write ( gameIDstring , 8 ) ;
2007-01-16 12:40:51 +00:00
debugC ( 5 , kDebugLevelMain | kDebugLevelSavegame , " Writing game id (%s, %s) " , gameIDstring , _game . id ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:41:50 +00:00
const char * tmp = getGameMD5 ( ) ;
2009-09-23 13:17:08 +00:00
// As reported in bug report #2849084 "AGI: Crash when saving fallback-matched game"
// getGameMD5 will return NULL for fallback matched games. Since there is also no
// filename available we can not compute any MD5 here either. Thus we will just set
// the MD5 sum in the savegame to all zero, when getGameMD5 returns NULL.
if ( ! tmp ) {
for ( i = 0 ; i < 32 ; + + i )
out - > writeByte ( 0 ) ;
} else {
for ( i = 0 ; i < 32 ; + + i )
out - > writeByte ( tmp [ i ] ) ;
}
2009-06-06 17:41:50 +00:00
2006-05-23 23:43:52 +00:00
for ( i = 0 ; i < MAX_FLAGS ; i + + )
2007-01-16 12:40:51 +00:00
out - > writeByte ( _game . flags [ i ] ) ;
2006-05-23 23:43:52 +00:00
for ( i = 0 ; i < MAX_VARS ; i + + )
2007-01-16 12:40:51 +00:00
out - > writeByte ( _game . vars [ i ] ) ;
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
out - > writeSint16BE ( ( int8 ) _game . horizon ) ;
out - > writeSint16BE ( ( int16 ) _game . lineStatus ) ;
out - > writeSint16BE ( ( int16 ) _game . lineUserInput ) ;
out - > writeSint16BE ( ( int16 ) _game . lineMinPrint ) ;
2007-01-12 02:29:20 +00:00
2007-01-16 12:40:51 +00:00
out - > writeSint16BE ( ( int16 ) _game . inputMode ) ;
out - > writeSint16BE ( ( int16 ) _game . lognum ) ;
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
out - > writeSint16BE ( ( int16 ) _game . playerControl ) ;
2008-09-30 12:27:38 +00:00
out - > writeSint16BE ( ( int16 ) shouldQuit ( ) ) ;
2007-01-16 12:40:51 +00:00
out - > writeSint16BE ( ( int16 ) _game . statusLine ) ;
out - > writeSint16BE ( ( int16 ) _game . clockEnabled ) ;
out - > writeSint16BE ( ( int16 ) _game . exitAllLogics ) ;
out - > writeSint16BE ( ( int16 ) _game . pictureShown ) ;
out - > writeSint16BE ( ( int16 ) _game . hasPrompt ) ;
out - > writeSint16BE ( ( int16 ) _game . gameFlags ) ;
2006-05-23 23:43:52 +00:00
2010-06-18 22:07:32 +00:00
out - > writeSint16BE ( _game . inputEnabled ) ;
2007-01-12 02:29:20 +00:00
for ( i = 0 ; i < _HEIGHT ; i + + )
2007-01-16 12:40:51 +00:00
out - > writeByte ( _game . priTable [ i ] ) ;
2007-09-19 08:40:12 +00:00
2007-01-16 12:40:51 +00:00
out - > writeSint16BE ( ( int16 ) _game . gfxMode ) ;
out - > writeByte ( _game . cursorChar ) ;
out - > writeSint16BE ( ( int16 ) _game . colorFg ) ;
out - > writeSint16BE ( ( int16 ) _game . colorBg ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// game.hires
// game.sbuf
// game.ego_words
// game.num_ego_words
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
out - > writeSint16BE ( ( int16 ) _game . numObjects ) ;
for ( i = 0 ; i < ( int16 ) _game . numObjects ; i + + )
out - > writeSint16BE ( ( int16 ) objectGetLocation ( i ) ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// game.ev_keyp
2006-05-23 23:43:52 +00:00
for ( i = 0 ; i < MAX_STRINGS ; i + + )
2007-01-16 12:40:51 +00:00
out - > write ( _game . strings [ i ] , MAX_STRINGLEN ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// record info about loaded resources
2006-05-23 23:43:52 +00:00
for ( i = 0 ; i < MAX_DIRS ; i + + ) {
2007-01-16 12:40:51 +00:00
out - > writeByte ( _game . dirLogic [ i ] . flags ) ;
out - > writeSint16BE ( ( int16 ) _game . logics [ i ] . sIP ) ;
out - > writeSint16BE ( ( int16 ) _game . logics [ i ] . cIP ) ;
2006-05-23 23:43:52 +00:00
}
for ( i = 0 ; i < MAX_DIRS ; i + + )
2007-01-16 12:40:51 +00:00
out - > writeByte ( _game . dirPic [ i ] . flags ) ;
2006-05-23 23:43:52 +00:00
for ( i = 0 ; i < MAX_DIRS ; i + + )
2007-01-16 12:40:51 +00:00
out - > writeByte ( _game . dirView [ i ] . flags ) ;
2006-05-23 23:43:52 +00:00
for ( i = 0 ; i < MAX_DIRS ; i + + )
2007-01-16 12:40:51 +00:00
out - > writeByte ( _game . dirSound [ i ] . flags ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// game.pictures
// game.logics
// game.views
// game.sounds
2006-05-23 23:43:52 +00:00
for ( i = 0 ; i < MAX_VIEWTABLE ; i + + ) {
2007-01-16 12:40:51 +00:00
VtEntry * v = & _game . viewTable [ i ] ;
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
out - > writeByte ( v - > stepTime ) ;
out - > writeByte ( v - > stepTimeCount ) ;
2007-01-12 02:29:20 +00:00
out - > writeByte ( v - > entry ) ;
2007-01-16 12:40:51 +00:00
out - > writeSint16BE ( v - > xPos ) ;
out - > writeSint16BE ( v - > yPos ) ;
out - > writeByte ( v - > currentView ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// v->view_data
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
out - > writeByte ( v - > currentLoop ) ;
out - > writeByte ( v - > numLoops ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// v->loop_data
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
out - > writeByte ( v - > currentCel ) ;
out - > writeByte ( v - > numCels ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// v->cel_data
// v->cel_data_2
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
out - > writeSint16BE ( v - > xPos2 ) ;
out - > writeSint16BE ( v - > yPos2 ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// v->s
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
out - > writeSint16BE ( v - > xSize ) ;
out - > writeSint16BE ( v - > ySize ) ;
out - > writeByte ( v - > stepSize ) ;
out - > writeByte ( v - > cycleTime ) ;
out - > writeByte ( v - > cycleTimeCount ) ;
2007-01-12 02:29:20 +00:00
out - > writeByte ( v - > direction ) ;
2006-05-23 23:43:52 +00:00
2007-01-12 02:29:20 +00:00
out - > writeByte ( v - > motion ) ;
out - > writeByte ( v - > cycle ) ;
out - > writeByte ( v - > priority ) ;
2006-05-23 23:43:52 +00:00
2007-01-12 02:29:20 +00:00
out - > writeUint16BE ( v - > flags ) ;
2006-05-23 23:43:52 +00:00
2007-01-12 02:29:20 +00:00
out - > writeByte ( v - > parm1 ) ;
out - > writeByte ( v - > parm2 ) ;
out - > writeByte ( v - > parm3 ) ;
out - > writeByte ( v - > parm4 ) ;
2006-05-23 23:43:52 +00:00
}
2009-06-06 17:39:13 +00:00
// Save image stack
2006-05-23 23:43:52 +00:00
2012-02-22 20:20:55 +01:00
for ( Common : : Stack < ImageStackElement > : : size_type j = 0 ; j < _imageStack . size ( ) ; + + j ) {
const ImageStackElement & ise = _imageStack [ j ] ;
2007-11-22 10:32:36 +00:00
out - > writeByte ( ise . type ) ;
out - > writeSint16BE ( ise . parm1 ) ;
out - > writeSint16BE ( ise . parm2 ) ;
out - > writeSint16BE ( ise . parm3 ) ;
out - > writeSint16BE ( ise . parm4 ) ;
out - > writeSint16BE ( ise . parm5 ) ;
out - > writeSint16BE ( ise . parm6 ) ;
out - > writeSint16BE ( ise . parm7 ) ;
2006-05-23 23:43:52 +00:00
}
2007-01-12 02:29:20 +00:00
out - > writeByte ( 0 ) ;
2007-03-22 22:03:21 +00:00
//Write which file number AGIPAL is using (0 if not being used)
2007-03-22 22:29:00 +00:00
out - > writeSint16BE ( _gfx - > getAGIPalFileNum ( ) ) ;
2007-03-22 22:03:21 +00:00
2007-02-17 18:55:51 +00:00
out - > finalize ( ) ;
2009-05-19 11:42:14 +00:00
if ( out - > err ( ) ) {
2011-09-25 21:25:25 +03:00
warning ( " Can't write file '%s'. (Disk full?) " , fileName . c_str ( ) ) ;
2007-08-04 06:18:28 +00:00
result = errIOError ;
} else
2011-09-25 22:23:22 +03:00
debugC ( 1 , kDebugLevelMain | kDebugLevelSavegame , " Saved game %s in file %s " , description . c_str ( ) , fileName . c_str ( ) ) ;
2007-09-19 08:40:12 +00:00
2007-01-12 02:29:20 +00:00
delete out ;
2011-09-25 21:25:25 +03:00
debugC ( 3 , kDebugLevelMain | kDebugLevelSavegame , " Closed %s " , fileName . c_str ( ) ) ;
2008-08-10 22:53:43 +00:00
_lastSaveTime = _system - > getMillis ( ) ;
2007-08-04 06:18:28 +00:00
return result ;
2006-05-23 23:43:52 +00:00
}
2011-09-26 09:27:15 +02:00
int AgiEngine : : loadGame ( const Common : : String & fileName , bool checkId ) {
2007-01-12 02:29:20 +00:00
char description [ 31 ] , saveVersion , loadId [ 8 ] ;
2007-01-16 12:40:51 +00:00
int i , vtEntries = MAX_VIEWTABLE ;
2006-05-23 23:43:52 +00:00
uint8 t ;
int16 parm [ 7 ] ;
2007-01-12 02:29:20 +00:00
Common : : InSaveFile * in ;
2006-05-23 23:43:52 +00:00
2011-09-25 21:25:25 +03:00
debugC ( 3 , kDebugLevelMain | kDebugLevelSavegame , " AgiEngine::loadGame(%s) " , fileName . c_str ( ) ) ;
2006-05-23 23:43:52 +00:00
2007-01-12 02:29:20 +00:00
if ( ! ( in = _saveFileMan - > openForLoading ( fileName ) ) ) {
2011-09-25 21:25:25 +03:00
warning ( " Can't open file '%s', game not loaded " , fileName . c_str ( ) ) ;
2007-01-16 12:40:51 +00:00
return errBadFileOpen ;
2007-01-12 02:29:20 +00:00
} else {
2011-09-25 21:25:25 +03:00
debugC ( 3 , kDebugLevelMain | kDebugLevelSavegame , " Successfully opened %s for reading " , fileName . c_str ( ) ) ;
2006-05-23 23:43:52 +00:00
}
2007-01-12 02:29:20 +00:00
uint32 typea = in - > readUint32BE ( ) ;
if ( typea = = AGIflag ) {
debugC ( 6 , kDebugLevelMain | kDebugLevelSavegame , " Has AGI flag, good start " ) ;
} else {
warning ( " This doesn't appear to be an AGI savegame, game not restored " ) ;
2007-09-19 08:40:12 +00:00
delete in ;
2007-01-16 12:40:51 +00:00
return errOK ;
2007-09-19 08:40:12 +00:00
}
2007-01-12 02:29:20 +00:00
in - > read ( description , 31 ) ;
2007-09-19 08:40:12 +00:00
2007-01-12 02:29:20 +00:00
debugC ( 6 , kDebugLevelMain | kDebugLevelSavegame , " Description is: %s " , description ) ;
2007-09-19 08:40:12 +00:00
2007-01-12 02:29:20 +00:00
saveVersion = in - > readByte ( ) ;
2008-11-10 19:02:47 +00:00
if ( saveVersion < 2 ) // is the save game pre-ScummVM?
2007-01-12 02:29:20 +00:00
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 ) ;
2008-11-10 19:02:47 +00:00
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 > = 4 ) {
// We don't need the thumbnail here, so just read it and discard it
2011-08-07 14:34:15 +02:00
Graphics : : skipThumbnail ( * in ) ;
2008-11-10 19:02:47 +00:00
in - > readUint32BE ( ) ; // save date
in - > readUint16BE ( ) ; // save time
2012-03-27 00:50:29 +03:00
if ( saveVersion > = 6 ) {
uint32 playTime = in - > readUint32BE ( ) ;
g_engine - > setTotalPlayTime ( playTime * 1000 ) ;
}
2008-11-10 19:02:47 +00:00
}
2010-06-18 22:07:32 +00:00
_game . state = ( State ) in - > readByte ( ) ;
2007-09-19 08:40:12 +00:00
2007-01-12 02:29:20 +00:00
in - > read ( loadId , 8 ) ;
2011-11-02 22:20:40 +00:00
if ( strcmp ( loadId , _game . id ) ! = 0 & & checkId ) {
2007-09-19 08:40:12 +00:00
delete in ;
2007-01-16 12:40:51 +00:00
warning ( " This save seems to be from a different AGI game (save from %s, running %s), not loaded " , loadId , _game . id ) ;
return errBadFileOpen ;
2006-05-23 23:43:52 +00:00
}
2009-01-07 20:18:38 +00:00
strncpy ( _game . id , loadId , 8 ) ;
2009-06-06 17:41:50 +00:00
if ( saveVersion > = 5 ) {
char md5 [ 32 + 1 ] ;
for ( i = 0 ; i < 32 ; i + + ) {
md5 [ i ] = in - > readByte ( ) ;
}
md5 [ i ] = 0 ; // terminate
2009-09-23 13:17:08 +00:00
// As noted above in AgiEngine::saveGame the MD5 sum field may be all zero
// when the save was made via a fallback matched game. In this case we will
// replace the MD5 sum with a nicer string, so that the user can easily see
// this fact in the debug output. The string saved in "md5" will never match
// any valid MD5 sum, thus it is safe to do that here.
if ( md5 [ 0 ] = = 0 )
strcpy ( md5 , " fallback matched " ) ;
debug ( 0 , " Saved game MD5: \" %s \" " , md5 ) ;
if ( ! getGameMD5 ( ) ) {
2010-09-18 10:55:16 +00:00
warning ( " Since your game was only detected via the fallback detector, there is no possibility to assure the save is compatible with your game version " ) ;
2009-06-06 17:41:50 +00:00
2009-09-23 13:17:08 +00:00
debug ( 0 , " The game used for saving is \" %s \" . " , md5 ) ;
2011-11-02 22:20:40 +00:00
} else if ( strcmp ( md5 , getGameMD5 ( ) ) ! = 0 ) {
2009-06-06 17:41:50 +00:00
warning ( " Game was saved with different gamedata - you may encounter problems " ) ;
2009-09-23 13:17:08 +00:00
debug ( 0 , " Your game is \" %s \" and save is \" %s \" . " , getGameMD5 ( ) , md5 ) ;
2009-06-06 17:41:50 +00:00
}
}
2010-01-25 01:39:44 +00:00
2006-05-23 23:43:52 +00:00
for ( i = 0 ; i < MAX_FLAGS ; i + + )
2007-01-16 12:40:51 +00:00
_game . flags [ i ] = in - > readByte ( ) ;
2006-05-23 23:43:52 +00:00
for ( i = 0 ; i < MAX_VARS ; i + + )
2007-01-16 12:40:51 +00:00
_game . vars [ i ] = in - > readByte ( ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:37:51 +00:00
setvar ( vFreePages , 180 ) ; // Set amount of free memory to realistic value (Overwriting the just loaded value)
2008-01-02 00:35:32 +00:00
2007-01-16 12:40:51 +00:00
_game . horizon = in - > readSint16BE ( ) ;
_game . lineStatus = in - > readSint16BE ( ) ;
_game . lineUserInput = in - > readSint16BE ( ) ;
_game . lineMinPrint = in - > readSint16BE ( ) ;
2007-09-19 08:40:12 +00:00
2009-06-06 17:39:13 +00:00
// These are never saved
2007-01-16 12:40:51 +00:00
_game . cursorPos = 0 ;
_game . inputBuffer [ 0 ] = 0 ;
_game . echoBuffer [ 0 ] = 0 ;
_game . keypress = 0 ;
2010-06-18 22:07:32 +00:00
_game . inputMode = ( InputMode ) in - > readSint16BE ( ) ;
2007-01-16 12:40:51 +00:00
_game . lognum = in - > readSint16BE ( ) ;
_game . playerControl = in - > readSint16BE ( ) ;
2008-07-07 23:24:12 +00:00
if ( in - > readSint16BE ( ) )
2008-07-09 02:27:05 +00:00
quitGame ( ) ;
2007-01-16 12:40:51 +00:00
_game . statusLine = in - > readSint16BE ( ) ;
_game . clockEnabled = in - > readSint16BE ( ) ;
_game . exitAllLogics = in - > readSint16BE ( ) ;
_game . pictureShown = in - > readSint16BE ( ) ;
_game . hasPrompt = in - > readSint16BE ( ) ;
_game . gameFlags = in - > readSint16BE ( ) ;
_game . inputEnabled = in - > readSint16BE ( ) ;
2006-05-23 23:43:52 +00:00
for ( i = 0 ; i < _HEIGHT ; i + + )
2007-01-16 12:40:51 +00:00
_game . priTable [ i ] = in - > readByte ( ) ;
2006-12-06 19:27:02 +00:00
2007-01-16 12:40:51 +00:00
if ( _game . hasWindow )
closeWindow ( ) ;
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
_game . msgBoxTicks = 0 ;
_game . block . active = false ;
2009-06-06 17:39:13 +00:00
// game.window - fixed by close_window()
// game.has_window - fixed by close_window()
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
_game . gfxMode = in - > readSint16BE ( ) ;
_game . cursorChar = in - > readByte ( ) ;
_game . colorFg = in - > readSint16BE ( ) ;
_game . colorBg = in - > readSint16BE ( ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// game.hires - rebuilt from image stack
// game.sbuf - rebuilt from image stack
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// game.ego_words - fixed by clean_input
// game.num_ego_words - fixed by clean_input
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
_game . numObjects = in - > readSint16BE ( ) ;
for ( i = 0 ; i < ( int16 ) _game . numObjects ; i + + )
objectSetLocation ( i , in - > readSint16BE ( ) ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// Those are not serialized
2006-05-23 23:43:52 +00:00
for ( i = 0 ; i < MAX_DIRS ; i + + ) {
2009-06-06 17:48:09 +00:00
_game . controllerOccured [ i ] = false ;
2006-05-23 23:43:52 +00:00
}
for ( i = 0 ; i < MAX_STRINGS ; i + + )
2007-01-16 12:40:51 +00:00
in - > read ( _game . strings [ i ] , MAX_STRINGLEN ) ;
2006-05-23 23:43:52 +00:00
for ( i = 0 ; i < MAX_DIRS ; i + + ) {
2007-01-12 02:29:20 +00:00
if ( in - > readByte ( ) & RES_LOADED )
agiLoadResource ( rLOGIC , i ) ;
2006-05-23 23:43:52 +00:00
else
2007-01-12 02:29:20 +00:00
agiUnloadResource ( rLOGIC , i ) ;
2007-01-16 12:40:51 +00:00
_game . logics [ i ] . sIP = in - > readSint16BE ( ) ;
_game . logics [ i ] . cIP = in - > readSint16BE ( ) ;
2006-05-23 23:43:52 +00:00
}
for ( i = 0 ; i < MAX_DIRS ; i + + ) {
2007-01-12 02:29:20 +00:00
if ( in - > readByte ( ) & RES_LOADED )
agiLoadResource ( rPICTURE , i ) ;
2006-05-23 23:43:52 +00:00
else
2007-01-12 02:29:20 +00:00
agiUnloadResource ( rPICTURE , i ) ;
2006-05-23 23:43:52 +00:00
}
for ( i = 0 ; i < MAX_DIRS ; i + + ) {
2007-01-12 02:29:20 +00:00
if ( in - > readByte ( ) & RES_LOADED )
agiLoadResource ( rVIEW , i ) ;
2006-05-23 23:43:52 +00:00
else
2007-01-12 02:29:20 +00:00
agiUnloadResource ( rVIEW , i ) ;
2006-05-23 23:43:52 +00:00
}
for ( i = 0 ; i < MAX_DIRS ; i + + ) {
2007-01-12 02:29:20 +00:00
if ( in - > readByte ( ) & RES_LOADED )
agiLoadResource ( rSOUND , i ) ;
2006-05-23 23:43:52 +00:00
else
2007-01-12 02:29:20 +00:00
agiUnloadResource ( rSOUND , i ) ;
2006-05-23 23:43:52 +00:00
}
2009-06-06 17:39:13 +00:00
// game.pictures - loaded above
// game.logics - loaded above
// game.views - loaded above
// game.sounds - loaded above
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
for ( i = 0 ; i < vtEntries ; i + + ) {
2007-02-03 21:37:52 +00:00
VtEntry * v = & _game . viewTable [ i ] ;
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
v - > stepTime = in - > readByte ( ) ;
v - > stepTimeCount = in - > readByte ( ) ;
2007-01-12 02:29:20 +00:00
v - > entry = in - > readByte ( ) ;
2007-01-16 12:40:51 +00:00
v - > xPos = in - > readSint16BE ( ) ;
v - > yPos = in - > readSint16BE ( ) ;
v - > currentView = in - > readByte ( ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// v->view_data - fixed below
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
v - > currentLoop = in - > readByte ( ) ;
v - > numLoops = in - > readByte ( ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// v->loop_data - fixed below
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
v - > currentCel = in - > readByte ( ) ;
v - > numCels = in - > readByte ( ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// v->cel_data - fixed below
// v->cel_data_2 - fixed below
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
v - > xPos2 = in - > readSint16BE ( ) ;
v - > yPos2 = in - > readSint16BE ( ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// v->s - fixed below
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
v - > xSize = in - > readSint16BE ( ) ;
v - > ySize = in - > readSint16BE ( ) ;
v - > stepSize = in - > readByte ( ) ;
v - > cycleTime = in - > readByte ( ) ;
v - > cycleTimeCount = in - > readByte ( ) ;
2007-01-12 02:29:20 +00:00
v - > direction = in - > readByte ( ) ;
2006-05-23 23:43:52 +00:00
2011-08-14 18:47:51 +01:00
v - > motion = ( MotionType ) in - > readByte ( ) ;
v - > cycle = ( CycleType ) in - > readByte ( ) ;
2007-01-12 02:29:20 +00:00
v - > priority = in - > readByte ( ) ;
2006-05-23 23:43:52 +00:00
2007-01-12 02:29:20 +00:00
v - > flags = in - > readUint16BE ( ) ;
2006-05-23 23:43:52 +00:00
2007-01-12 02:29:20 +00:00
v - > parm1 = in - > readByte ( ) ;
v - > parm2 = in - > readByte ( ) ;
v - > parm3 = in - > readByte ( ) ;
v - > parm4 = in - > readByte ( ) ;
2006-05-23 23:43:52 +00:00
}
2007-01-16 12:40:51 +00:00
for ( i = vtEntries ; i < MAX_VIEWTABLE ; i + + ) {
2007-02-03 21:37:52 +00:00
memset ( & _game . viewTable [ i ] , 0 , sizeof ( VtEntry ) ) ;
2006-05-23 23:43:52 +00:00
}
2009-06-06 17:39:13 +00:00
// Fix some pointers in viewtable
2006-05-23 23:43:52 +00:00
for ( i = 0 ; i < MAX_VIEWTABLE ; i + + ) {
2007-02-03 21:37:52 +00:00
VtEntry * v = & _game . viewTable [ i ] ;
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
if ( _game . dirView [ v - > currentView ] . offset = = _EMPTY )
2006-05-23 23:43:52 +00:00
continue ;
2007-01-16 12:40:51 +00:00
if ( ! ( _game . dirView [ v - > currentView ] . flags & RES_LOADED ) )
agiLoadResource ( rVIEW , v - > currentView ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
setView ( v , v - > currentView ) ; // Fix v->view_data
setLoop ( v , v - > currentLoop ) ; // Fix v->loop_data
setCel ( v , v - > currentCel ) ; // Fix v->cel_data
2007-01-16 12:40:51 +00:00
v - > celData2 = v - > celData ;
2009-06-06 17:39:13 +00:00
v - > s = NULL ; // not sure if it is used...
2006-05-23 23:43:52 +00:00
}
2007-01-16 12:40:51 +00:00
_sprites - > eraseBoth ( ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// Clear input line
2006-12-06 19:27:02 +00:00
_gfx - > clearScreen ( 0 ) ;
2007-01-16 12:40:51 +00:00
writeStatus ( ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
// Recreate background from saved image stack
2007-01-16 12:40:51 +00:00
clearImageStack ( ) ;
2007-01-12 02:29:20 +00:00
while ( ( t = in - > readByte ( ) ) ! = 0 ) {
2006-05-23 23:43:52 +00:00
for ( i = 0 ; i < 7 ; i + + )
2007-01-12 02:29:20 +00:00
parm [ i ] = in - > readSint16BE ( ) ;
2007-01-16 12:40:51 +00:00
replayImageStackCall ( t , parm [ 0 ] , parm [ 1 ] , parm [ 2 ] ,
2006-05-23 23:43:52 +00:00
parm [ 3 ] , parm [ 4 ] , parm [ 5 ] , parm [ 6 ] ) ;
}
2009-06-06 17:39:13 +00:00
// Load AGIPAL Data
2007-03-22 22:29:00 +00:00
if ( saveVersion > = 3 )
2007-03-22 22:03:21 +00:00
_gfx - > setAGIPal ( in - > readSint16BE ( ) ) ;
2007-09-19 08:40:12 +00:00
delete in ;
2011-09-25 21:25:25 +03:00
debugC ( 3 , kDebugLevelMain | kDebugLevelSavegame , " Closed %s " , fileName . c_str ( ) ) ;
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
setflag ( fRestoreJustRan , true ) ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:39:13 +00:00
_game . hasPrompt = 0 ; // force input line repaint if necessary
2007-01-16 12:40:51 +00:00
cleanInput ( ) ;
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
_sprites - > eraseBoth ( ) ;
_sprites - > blitBoth ( ) ;
_sprites - > commitBoth ( ) ;
_picture - > showPic ( ) ;
2006-12-06 19:27:02 +00:00
_gfx - > doUpdate ( ) ;
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
return errOK ;
2006-05-23 23:43:52 +00:00
}
2007-03-12 08:43:13 +00:00
# define NUM_SLOTS 100
# define NUM_VISIBLE_SLOTS 12
2006-05-23 23:43:52 +00:00
2011-09-26 09:27:15 +02:00
Common : : String AgiEngine : : getSavegameFilename ( int num ) const {
2011-06-02 19:46:55 +01:00
Common : : String saveLoadSlot = _targetName ;
saveLoadSlot + = Common : : String : : format ( " .%.3d " , num ) ;
2011-09-25 21:25:25 +03:00
return saveLoadSlot ;
2007-01-12 02:29:20 +00:00
}
2007-03-12 08:43:13 +00:00
2007-03-12 19:19:30 +00:00
void AgiEngine : : getSavegameDescription ( int num , char * buf , bool showEmpty ) {
2007-03-12 08:43:13 +00:00
Common : : InSaveFile * in ;
2011-09-25 21:25:25 +03:00
Common : : String fileName = getSavegameFilename ( num ) ;
2007-09-19 08:40:12 +00:00
2007-03-12 08:43:13 +00:00
debugC ( 4 , kDebugLevelMain | kDebugLevelSavegame , " Current game id is %s " , _targetName . c_str ( ) ) ;
2011-09-25 19:33:01 +03:00
2007-03-12 08:43:13 +00:00
if ( ! ( in = _saveFileMan - > openForLoading ( fileName ) ) ) {
2011-09-25 21:25:25 +03:00
debugC ( 4 , kDebugLevelMain | kDebugLevelSavegame , " File %s does not exist " , fileName . c_str ( ) ) ;
2009-06-06 17:39:13 +00:00
2007-03-12 19:19:30 +00:00
if ( showEmpty )
strcpy ( buf , " (empty slot) " ) ;
else
* buf = 0 ;
2007-03-12 08:43:13 +00:00
} else {
2011-09-25 21:25:25 +03:00
debugC ( 4 , kDebugLevelMain | kDebugLevelSavegame , " Successfully opened %s for reading " , fileName . c_str ( ) ) ;
2009-06-06 17:39:13 +00:00
2007-03-12 08:43:13 +00:00
uint32 type = in - > readUint32BE ( ) ;
2009-06-06 17:39:13 +00:00
2007-03-12 08:43:13 +00:00
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) " ) ;
2007-09-19 08:40:12 +00:00
}
2007-03-12 08:43:13 +00:00
delete in ;
}
}
2007-09-19 08:40:12 +00:00
2007-01-12 02:29:20 +00:00
int AgiEngine : : selectSlot ( ) {
2006-05-23 23:43:52 +00:00
int i , key , active = 0 ;
int rc = - 1 ;
2009-06-06 17:39:13 +00:00
int hm = 1 , vm = 3 ; // box margins
2007-02-10 16:34:45 +00:00
int xmin , xmax , slotClicked ;
2007-03-12 08:43:13 +00:00
char desc [ NUM_VISIBLE_SLOTS ] [ 40 ] ;
2011-04-14 14:34:28 +02:00
int textCenter , buttonLength , buttonX [ 2 ] , buttonY ;
2007-02-10 16:34:45 +00:00
const char * buttonText [ ] = { " OK " , " Cancel " , NULL } ;
2006-05-23 23:43:52 +00:00
2009-06-06 17:42:37 +00:00
_noSaveLoadAllowed = true ;
2009-05-21 21:51:50 +00:00
2007-03-12 08:43:13 +00:00
for ( i = 0 ; i < NUM_VISIBLE_SLOTS ; i + + ) {
2009-01-27 19:59:50 +00:00
getSavegameDescription ( _firstSlot + i , desc [ i ] ) ;
2006-05-23 23:43:52 +00:00
}
2011-04-14 14:34:28 +02:00
textCenter = GFX_WIDTH / CHAR_LINES / 2 ;
2007-02-10 16:34:45 +00:00
buttonLength = 6 ;
2011-04-14 14:34:28 +02:00
buttonX [ 0 ] = ( textCenter - 3 * buttonLength / 2 ) * CHAR_COLS ;
buttonX [ 1 ] = ( textCenter + buttonLength / 2 ) * CHAR_COLS ;
2007-02-10 16:34:45 +00:00
buttonY = ( vm + 17 ) * CHAR_LINES ;
2007-09-19 08:40:12 +00:00
2007-02-10 16:34:45 +00:00
for ( i = 0 ; i < 2 ; i + + )
2007-07-10 18:32:24 +00:00
_gfx - > drawCurrentStyleButton ( buttonX [ i ] , buttonY , buttonText [ i ] , false , false , i = = 0 ) ;
2007-02-10 16:34:45 +00:00
2007-03-17 16:08:29 +00:00
AllowSyntheticEvents on ( this ) ;
2007-07-06 17:50:27 +00:00
int oldFirstSlot = _firstSlot + 1 ;
int oldActive = active + 1 ;
2011-12-10 20:12:35 +00:00
bool exitSelectSlot = false ;
while ( ! exitSelectSlot & & ! ( shouldQuit ( ) | | _restartGame ) ) {
2007-09-08 11:17:13 +00:00
int sbPos = 0 ;
2007-03-12 08:43:13 +00:00
2008-12-19 17:21:55 +00:00
// 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 ;
}
2007-07-06 17:50:27 +00:00
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 ,
2011-04-14 14:12:27 +02:00
( 40 - 2 * hm ) - 1 , i = = active ? MSG_BOX_COLOR : MSG_BOX_TEXT ,
i = = active ? MSG_BOX_TEXT : MSG_BOX_COLOR ) ;
2007-07-06 17:50:27 +00:00
}
2007-03-12 08:43:13 +00:00
2007-07-06 17:50:27 +00:00
char upArrow [ ] = " ^ " ;
char downArrow [ ] = " v " ;
char scrollBar [ ] = " " ;
for ( i = 1 ; i < NUM_VISIBLE_SLOTS - 1 ; i + + )
2011-04-14 14:12:27 +02:00
printText ( scrollBar , 35 , hm + 1 , vm + 4 + i , 1 , MSG_BOX_COLOR , 7 , true ) ;
2007-03-12 08:43:13 +00:00
2007-07-06 17:50:27 +00:00
printText ( upArrow , 35 , hm + 1 , vm + 4 , 1 , 8 , 7 ) ;
printText ( downArrow , 35 , hm + 1 , vm + 4 + NUM_VISIBLE_SLOTS - 1 , 1 , 8 , 7 ) ;
2011-04-14 14:12:27 +02:00
printText ( scrollBar , 35 , hm + 1 , vm + 4 + sbPos , 1 , MSG_BOX_COLOR , MSG_BOX_TEXT ) ;
2007-03-12 08:43:13 +00:00
2007-07-06 17:50:27 +00:00
oldActive = active ;
oldFirstSlot = _firstSlot ;
}
2007-09-19 08:40:12 +00:00
2009-06-06 17:45:52 +00:00
pollTimer ( ) ;
2007-01-16 12:40:51 +00:00
key = doPollKeyboard ( ) ;
2009-05-19 21:09:43 +00:00
// 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 ;
2011-12-10 20:12:35 +00:00
exitSelectSlot = true ;
2009-05-19 21:09:43 +00:00
}
2011-12-10 20:12:35 +00:00
if ( ! exitSelectSlot ) {
switch ( key ) {
case KEY_ENTER :
2007-02-10 16:34:45 +00:00
rc = active ;
strncpy ( _game . strings [ MAX_STRINGS ] , desc [ i ] , MAX_STRINGLEN ) ;
2011-12-10 20:12:35 +00:00
debugC ( 8 , kDebugLevelMain | kDebugLevelInput , " Button pressed: %d " , rc ) ;
exitSelectSlot = true ;
break ;
case KEY_ESCAPE :
2007-02-10 16:34:45 +00:00
rc = - 1 ;
2011-12-10 20:12:35 +00:00
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 ) ;
}
}
2007-03-12 08:43:13 +00:00
}
2011-12-10 20:12:35 +00:00
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 ) {
2007-03-12 08:43:13 +00:00
_firstSlot + + ;
for ( i = 1 ; i < NUM_VISIBLE_SLOTS ; i + + )
memcpy ( desc [ i - 1 ] , desc [ i ] , sizeof ( desc [ 0 ] ) ) ;
2009-01-27 19:59:50 +00:00
getSavegameDescription ( _firstSlot + NUM_VISIBLE_SLOTS - 1 , desc [ NUM_VISIBLE_SLOTS - 1 ] ) ;
2007-03-12 08:43:13 +00:00
}
2011-12-10 20:12:35 +00:00
break ;
case WHEEL_UP :
2007-03-12 08:43:13 +00:00
if ( _firstSlot > 0 ) {
_firstSlot - - ;
for ( i = NUM_VISIBLE_SLOTS - 1 ; i > 0 ; i - - )
memcpy ( desc [ i ] , desc [ i - 1 ] , sizeof ( desc [ 0 ] ) ) ;
2009-01-27 19:59:50 +00:00
getSavegameDescription ( _firstSlot , desc [ 0 ] ) ;
2007-03-12 08:43:13 +00:00
}
2011-12-10 20:12:35 +00:00
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 ;
2007-03-12 08:43:13 +00:00
}
2006-05-23 23:43:52 +00:00
}
2006-12-06 19:27:02 +00:00
_gfx - > doUpdate ( ) ;
2006-05-23 23:43:52 +00:00
}
2007-01-16 12:40:51 +00:00
closeWindow ( ) ;
2009-05-19 21:09:43 +00:00
2009-06-06 17:42:37 +00:00
_noSaveLoadAllowed = false ;
2009-05-21 21:51:50 +00:00
2006-05-23 23:43:52 +00:00
return rc ;
}
2012-03-27 00:21:16 +03:00
int AgiEngine : : scummVMSaveLoadDialog ( bool isSave ) {
const EnginePlugin * plugin = NULL ;
EngineMan . findGame ( ConfMan . get ( " gameid " ) , & plugin ) ;
GUI : : SaveLoadChooser * dialog ;
Common : : String desc ;
int slot ;
if ( isSave ) {
dialog = new GUI : : SaveLoadChooser ( _ ( " Save game: " ) , _ ( " Save " ) ) ;
dialog - > setSaveMode ( true ) ;
slot = dialog - > runModalWithPluginAndTarget ( plugin , ConfMan . getActiveDomainName ( ) ) ;
desc = dialog - > getResultString ( ) ;
if ( desc . empty ( ) ) {
// create our own description for the saved game, the user didnt enter it
# if defined(USE_SAVEGAME_TIMESTAMP)
TimeDate curTime ;
g_system - > getTimeAndDate ( curTime ) ;
curTime . tm_year + = 1900 ; // fixup year
curTime . tm_mon + + ; // fixup month
desc = Common : : String : : format ( " %04d.%02d.%02d / %02d:%02d:%02d " , curTime . tm_year , curTime . tm_mon , curTime . tm_mday , curTime . tm_hour , curTime . tm_min , curTime . tm_sec ) ;
# else
desc = Common : : String : : format ( " Save %d " , slot + 1 ) ;
# endif
}
if ( desc . size ( ) > 28 )
desc = Common : : String ( desc . c_str ( ) , 28 ) ;
} else {
dialog = new GUI : : SaveLoadChooser ( _ ( " Restore game: " ) , _ ( " Restore " ) ) ;
dialog - > setSaveMode ( false ) ;
slot = dialog - > runModalWithPluginAndTarget ( plugin , ConfMan . getActiveDomainName ( ) ) ;
}
delete dialog ;
2012-05-02 23:51:02 +03:00
if ( slot < 0 )
return true ;
2012-03-27 00:21:16 +03:00
if ( isSave )
return doSave ( slot , desc ) ;
else
return doLoad ( slot , false ) ;
}
int AgiEngine : : doSave ( int slot , const Common : : String & desc ) {
Common : : String fileName = getSavegameFilename ( slot ) ;
debugC ( 8 , kDebugLevelMain | kDebugLevelResources , " file is [%s] " , fileName . c_str ( ) ) ;
// Make sure all graphics was blitted to screen. This fixes bug
// #2960567: "AGI: Ego partly erased in Load/Save thumbnails"
_gfx - > doUpdate ( ) ;
return saveGame ( fileName , desc ) ;
}
int AgiEngine : : doLoad ( int slot , bool showMessages ) {
Common : : String fileName = getSavegameFilename ( slot ) ;
debugC ( 8 , kDebugLevelMain | kDebugLevelResources , " file is [%s] " , fileName . c_str ( ) ) ;
_sprites - > eraseBoth ( ) ;
_sound - > stopSound ( ) ;
closeWindow ( ) ;
int result = loadGame ( fileName ) ;
if ( result = = errOK ) {
if ( showMessages )
messageBox ( " Game restored. " ) ;
_game . exitAllLogics = 1 ;
_menu - > enableAll ( ) ;
} else {
if ( showMessages )
messageBox ( " Error restoring game. " ) ;
}
return result ;
}
2007-01-12 02:29:20 +00:00
int AgiEngine : : saveGameDialog ( ) {
2012-03-27 00:21:16 +03:00
if ( ! ConfMan . getBool ( " originalsaveload " ) )
return scummVMSaveLoadDialog ( true ) ;
2006-05-23 23:43:52 +00:00
char * desc ;
2006-05-24 19:51:37 +00:00
const char * buttons [ ] = { " Do as I say! " , " I regret " , NULL } ;
2006-05-23 23:43:52 +00:00
char dstr [ 200 ] ;
int rc , slot = 0 ;
2007-09-19 08:40:12 +00:00
int hm , vm , hp , vp ;
2006-05-23 23:43:52 +00:00
int w ;
2007-03-12 08:43:13 +00:00
hm = 1 ;
2006-05-23 23:43:52 +00:00
vm = 3 ;
hp = hm * CHAR_COLS ;
vp = vm * CHAR_LINES ;
w = ( 40 - 2 * hm ) - 1 ;
2008-08-10 22:53:43 +00:00
do {
drawWindow ( hp , vp , GFX_WIDTH - hp , GFX_HEIGHT - vp ) ;
printText ( " Select a slot in which you wish to \n save the game: " ,
2011-04-14 14:12:27 +02:00
0 , hm + 1 , vm + 1 , w , MSG_BOX_TEXT , MSG_BOX_COLOR ) ;
2008-08-10 22:53:43 +00:00
slot = selectSlot ( ) ;
2010-10-03 18:38:38 +00:00
if ( slot + _firstSlot = = 0 )
2008-08-10 22:53:43 +00:00
messageBox ( " That slot is for Autosave only. " ) ;
else if ( slot < 0 )
return errOK ;
2010-10-03 18:38:38 +00:00
} while ( slot + _firstSlot = = 0 ) ;
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
drawWindow ( hp , vp + 5 * CHAR_LINES , GFX_WIDTH - hp ,
2006-05-23 23:43:52 +00:00
GFX_HEIGHT - vp - 9 * CHAR_LINES ) ;
2007-01-16 12:40:51 +00:00
printText ( " Enter a description for this game: " ,
2011-04-14 14:12:27 +02:00
0 , hm + 1 , vm + 6 , w , MSG_BOX_TEXT , MSG_BOX_COLOR ) ;
2006-12-06 19:27:02 +00:00
_gfx - > drawRectangle ( 3 * CHAR_COLS , 11 * CHAR_LINES - 1 ,
2006-05-23 23:43:52 +00:00
37 * CHAR_COLS , 12 * CHAR_LINES , MSG_BOX_TEXT ) ;
2006-12-06 19:27:02 +00:00
_gfx - > flushBlock ( 3 * CHAR_COLS , 11 * CHAR_LINES - 1 ,
2006-05-23 23:43:52 +00:00
37 * CHAR_COLS , 12 * CHAR_LINES ) ;
2007-03-12 19:19:30 +00:00
// 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 ;
2009-01-27 19:59:50 +00:00
getSavegameDescription ( _firstSlot + slot , name , false ) ;
2007-03-12 19:19:30 +00:00
for ( numChars = 0 ; numChars < 28 & & name [ numChars ] ; numChars + + )
handleGetstring ( name [ numChars ] ) ;
2011-04-14 14:12:27 +02:00
_gfx - > printCharacter ( numChars + 3 , 11 , _game . cursorChar , MSG_BOX_COLOR , MSG_BOX_TEXT ) ;
2006-05-23 23:43:52 +00:00
do {
2007-01-16 12:40:51 +00:00
mainCycle ( ) ;
} while ( _game . inputMode = = INPUT_GETSTRING ) ;
closeWindow ( ) ;
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
desc = _game . strings [ MAX_STRINGS ] ;
2006-05-23 23:43:52 +00:00
sprintf ( dstr , " Are you sure you want to save the game "
2009-01-27 19:59:50 +00:00
" described as: \n \n %s \n \n in slot %d? \n \n \n " , desc , _firstSlot + slot ) ;
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
rc = selectionBox ( dstr , buttons ) ;
2006-05-23 23:43:52 +00:00
if ( rc ! = 0 ) {
2007-01-16 12:40:51 +00:00
messageBox ( " Game NOT saved. " ) ;
return errOK ;
2006-05-23 23:43:52 +00:00
}
2012-03-27 00:21:16 +03:00
int result = doSave ( _firstSlot + slot , desc ) ;
2006-05-23 23:43:52 +00:00
2007-08-04 06:18:28 +00:00
if ( result = = errOK )
messageBox ( " Game saved. " ) ;
else
messageBox ( " Error saving game. " ) ;
2006-05-23 23:43:52 +00:00
2007-08-04 06:18:28 +00:00
return result ;
2006-05-23 23:43:52 +00:00
}
2007-01-12 02:29:20 +00:00
int AgiEngine : : saveGameSimple ( ) {
2012-03-27 00:21:16 +03:00
if ( ! ConfMan . getBool ( " originalsaveload " ) )
return scummVMSaveLoadDialog ( true ) ;
2011-09-25 21:25:25 +03:00
Common : : String fileName = getSavegameFilename ( 0 ) ;
2011-09-25 19:33:01 +03:00
2011-09-26 09:27:15 +02:00
int result = saveGame ( fileName , " Default savegame " ) ;
2007-08-04 06:18:28 +00:00
if ( result ! = errOK )
messageBox ( " Error saving game. " ) ;
return result ;
2006-05-23 23:43:52 +00:00
}
2007-01-12 02:29:20 +00:00
int AgiEngine : : loadGameDialog ( ) {
2012-03-27 00:21:16 +03:00
if ( ! ConfMan . getBool ( " originalsaveload " ) )
return scummVMSaveLoadDialog ( false ) ;
int slot = 0 ;
2009-06-06 17:39:13 +00:00
int hm , vm , hp , vp ; // box margins
2006-05-23 23:43:52 +00:00
int w ;
2007-03-12 08:43:13 +00:00
hm = 1 ;
2006-05-23 23:43:52 +00:00
vm = 3 ;
hp = hm * CHAR_COLS ;
vp = vm * CHAR_LINES ;
w = ( 40 - 2 * hm ) - 1 ;
2007-01-16 12:40:51 +00:00
_sprites - > eraseBoth ( ) ;
_sound - > stopSound ( ) ;
2006-05-23 23:43:52 +00:00
2007-01-16 12:40:51 +00:00
drawWindow ( hp , vp , GFX_WIDTH - hp , GFX_HEIGHT - vp ) ;
printText ( " Select a game which you wish to \n restore: " ,
2011-04-14 14:12:27 +02:00
0 , hm + 1 , vm + 1 , w , MSG_BOX_TEXT , MSG_BOX_COLOR ) ;
2006-05-23 23:43:52 +00:00
2007-01-12 02:29:20 +00:00
slot = selectSlot ( ) ;
2006-05-23 23:43:52 +00:00
if ( slot < 0 ) {
2009-05-19 21:09:43 +00:00
if ( slot = = - 1 ) // slot = -2 when GMM was launched
messageBox ( " Game NOT restored. " ) ;
2007-01-16 12:40:51 +00:00
return errOK ;
2006-05-23 23:43:52 +00:00
}
2012-03-27 00:21:16 +03:00
return doLoad ( _firstSlot + slot , true ) ;
2007-01-12 02:29:20 +00:00
}
int AgiEngine : : loadGameSimple ( ) {
2012-03-27 00:21:16 +03:00
if ( ! ConfMan . getBool ( " originalsaveload " ) )
return scummVMSaveLoadDialog ( false ) ;
else
return doLoad ( 0 , true ) ;
2006-05-23 23:43:52 +00:00
}
2009-07-22 15:55:33 +00:00
void AgiEngine : : recordImageStackCall ( uint8 type , int16 p1 , int16 p2 , int16 p3 ,
int16 p4 , int16 p5 , int16 p6 , int16 p7 ) {
ImageStackElement pnew ;
pnew . type = type ;
pnew . pad = 0 ;
pnew . parm1 = p1 ;
pnew . parm2 = p2 ;
pnew . parm3 = p3 ;
pnew . parm4 = p4 ;
pnew . parm5 = p5 ;
pnew . parm6 = p6 ;
pnew . parm7 = p7 ;
_imageStack . push ( pnew ) ;
}
void AgiEngine : : replayImageStackCall ( uint8 type , int16 p1 , int16 p2 , int16 p3 ,
int16 p4 , int16 p5 , int16 p6 , int16 p7 ) {
switch ( type ) {
case ADD_PIC :
debugC ( 8 , kDebugLevelMain , " --- decoding picture %d --- " , p1 ) ;
agiLoadResource ( rPICTURE , p1 ) ;
_picture - > decodePicture ( p1 , p2 , p3 ! = 0 ) ;
break ;
case ADD_VIEW :
agiLoadResource ( rVIEW , p1 ) ;
_sprites - > addToPic ( p1 , p2 , p3 , p4 , p5 , p6 , p7 ) ;
break ;
}
}
2009-11-02 21:54:57 +00:00
void AgiEngine : : clearImageStack ( ) {
2009-07-22 15:55:33 +00:00
_imageStack . clear ( ) ;
}
2009-11-02 21:54:57 +00:00
void AgiEngine : : releaseImageStack ( ) {
2009-07-22 15:55:33 +00:00
_imageStack . clear ( ) ;
}
void AgiEngine : : checkQuickLoad ( ) {
if ( ConfMan . hasKey ( " save_slot " ) ) {
2011-09-26 09:27:15 +02:00
Common : : String saveNameBuffer = getSavegameFilename ( ConfMan . getInt ( " save_slot " ) ) ;
2009-07-22 15:55:33 +00:00
2009-08-03 17:18:18 +00:00
_sprites - > eraseBoth ( ) ;
_sound - > stopSound ( ) ;
2011-09-26 09:27:15 +02:00
if ( loadGame ( saveNameBuffer , false ) = = errOK ) { // Do not check game id
2009-07-22 15:55:33 +00:00
_game . exitAllLogics = 1 ;
_menu - > enableAll ( ) ;
}
}
}
2009-08-03 17:18:18 +00:00
Common : : Error AgiEngine : : loadGameState ( int slot ) {
2011-09-26 09:27:15 +02:00
Common : : String saveLoadSlot = getSavegameFilename ( slot ) ;
2009-08-03 17:18:18 +00:00
_sprites - > eraseBoth ( ) ;
_sound - > stopSound ( ) ;
if ( loadGame ( saveLoadSlot ) = = errOK ) {
_game . exitAllLogics = 1 ;
_menu - > enableAll ( ) ;
return Common : : kNoError ;
} else {
return Common : : kUnknownError ;
}
}
2011-06-02 14:11:38 +02:00
Common : : Error AgiEngine : : saveGameState ( int slot , const Common : : String & desc ) {
2011-09-26 09:27:15 +02:00
Common : : String saveLoadSlot = getSavegameFilename ( slot ) ;
if ( saveGame ( saveLoadSlot , desc ) = = errOK )
2009-08-03 17:18:18 +00:00
return Common : : kNoError ;
else
return Common : : kUnknownError ;
}
2007-01-16 12:40:51 +00:00
} // End of namespace Agi