2007-05-30 21:56:52 +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.
|
2006-12-19 01:11:41 +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.
|
2014-02-18 01:34:17 +00:00
|
|
|
*
|
2006-12-19 01:11:41 +00:00
|
|
|
* 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.
|
2014-02-18 01:34:17 +00:00
|
|
|
*
|
2006-12-19 01:11:41 +00:00
|
|
|
* 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 "base/plugins.h"
|
|
|
|
|
2009-01-29 22:13:01 +00:00
|
|
|
#include "engines/advancedDetector.h"
|
2007-09-01 21:22:26 +00:00
|
|
|
#include "common/config-manager.h"
|
2006-12-19 01:11:41 +00:00
|
|
|
#include "common/file.h"
|
2011-05-25 13:27:35 +00:00
|
|
|
#include "common/md5.h"
|
2010-01-03 20:15:44 +00:00
|
|
|
#include "common/savefile.h"
|
2011-04-24 08:34:27 +00:00
|
|
|
#include "common/textconsole.h"
|
2012-03-26 21:21:16 +00:00
|
|
|
#include "common/translation.h"
|
|
|
|
|
2008-11-10 19:02:47 +00:00
|
|
|
#include "graphics/thumbnail.h"
|
2010-11-19 01:37:04 +00:00
|
|
|
#include "graphics/surface.h"
|
2006-12-19 01:11:41 +00:00
|
|
|
|
|
|
|
#include "agi/agi.h"
|
2007-09-18 16:20:44 +00:00
|
|
|
#include "agi/preagi.h"
|
2011-08-15 15:22:26 +00:00
|
|
|
#include "agi/preagi_mickey.h"
|
2011-08-15 15:33:20 +00:00
|
|
|
#include "agi/preagi_troll.h"
|
2011-08-15 04:21:19 +00:00
|
|
|
#include "agi/preagi_winnie.h"
|
2007-06-14 12:06:12 +00:00
|
|
|
#include "agi/wagparser.h"
|
2006-12-19 01:11:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
namespace Agi {
|
2007-01-21 20:24:38 +00:00
|
|
|
|
|
|
|
struct AGIGameDescription {
|
2009-01-29 22:13:01 +00:00
|
|
|
ADGameDescription desc;
|
2007-01-21 20:24:38 +00:00
|
|
|
|
2007-05-06 14:36:02 +00:00
|
|
|
int gameID;
|
2007-01-21 20:24:38 +00:00
|
|
|
int gameType;
|
|
|
|
uint32 features;
|
|
|
|
uint16 version;
|
|
|
|
};
|
|
|
|
|
2007-09-01 14:58:46 +00:00
|
|
|
uint32 AgiBase::getGameID() const {
|
2007-05-06 14:36:02 +00:00
|
|
|
return _gameDescription->gameID;
|
|
|
|
}
|
|
|
|
|
2007-09-01 14:58:46 +00:00
|
|
|
uint32 AgiBase::getFeatures() const {
|
2009-06-06 17:44:46 +00:00
|
|
|
return _gameFeatures;
|
2007-01-21 20:24:38 +00:00
|
|
|
}
|
|
|
|
|
2007-09-01 14:58:46 +00:00
|
|
|
Common::Platform AgiBase::getPlatform() const {
|
2007-02-10 17:10:55 +00:00
|
|
|
return _gameDescription->desc.platform;
|
|
|
|
}
|
|
|
|
|
2007-09-23 16:43:43 +00:00
|
|
|
Common::Language AgiBase::getLanguage() const {
|
|
|
|
return _gameDescription->desc.language;
|
|
|
|
}
|
|
|
|
|
2007-09-01 14:58:46 +00:00
|
|
|
uint16 AgiBase::getVersion() const {
|
2009-06-06 17:45:52 +00:00
|
|
|
return _gameVersion;
|
2007-01-21 20:24:38 +00:00
|
|
|
}
|
|
|
|
|
2007-09-01 14:58:46 +00:00
|
|
|
uint16 AgiBase::getGameType() const {
|
|
|
|
return _gameDescription->gameType;
|
|
|
|
}
|
|
|
|
|
2009-06-06 17:41:50 +00:00
|
|
|
const char *AgiBase::getGameMD5() const {
|
|
|
|
return _gameDescription->desc.filesDescriptions[0].md5;
|
|
|
|
}
|
|
|
|
|
2009-11-02 21:54:57 +00:00
|
|
|
void AgiBase::initFeatures() {
|
2009-06-06 17:44:46 +00:00
|
|
|
_gameFeatures = _gameDescription->features;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AgiBase::setFeature(uint32 feature) {
|
|
|
|
_gameFeatures |= feature;
|
|
|
|
}
|
|
|
|
|
2009-06-06 17:45:52 +00:00
|
|
|
void AgiBase::setVersion(uint16 version) {
|
|
|
|
_gameVersion = version;
|
|
|
|
}
|
|
|
|
|
2009-11-02 21:54:57 +00:00
|
|
|
void AgiBase::initVersion() {
|
2009-06-06 17:45:52 +00:00
|
|
|
_gameVersion = _gameDescription->version;
|
|
|
|
}
|
|
|
|
|
2011-08-13 22:25:19 +00:00
|
|
|
const char *AgiBase::getDiskName(uint16 id) {
|
|
|
|
for (int i = 0; _gameDescription->desc.filesDescriptions[i].fileName != NULL; i++)
|
|
|
|
if (_gameDescription->desc.filesDescriptions[i].fileType == id)
|
|
|
|
return _gameDescription->desc.filesDescriptions[i].fileName;
|
2011-06-04 15:12:26 +00:00
|
|
|
|
2011-08-13 22:25:19 +00:00
|
|
|
return "";
|
|
|
|
}
|
2011-06-04 15:12:26 +00:00
|
|
|
|
2006-12-19 01:11:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static const PlainGameDescriptor agiGames[] = {
|
2007-01-29 23:25:51 +00:00
|
|
|
{"agi", "Sierra AGI game"},
|
|
|
|
{"agi-fanmade", "Fanmade AGI game"},
|
|
|
|
{"agidemo", "AGI Demo"},
|
2007-02-04 13:40:11 +00:00
|
|
|
{"bc", "The Black Cauldron"},
|
2007-01-30 00:22:58 +00:00
|
|
|
{"caitlyn", "Caitlyn's Destiny"},
|
2007-01-29 23:25:51 +00:00
|
|
|
{"ddp", "Donald Duck's Playground"},
|
2007-02-04 13:40:11 +00:00
|
|
|
{"goldrush", "Gold Rush!"},
|
|
|
|
{"kq1", "King's Quest I: Quest for the Crown"},
|
|
|
|
{"kq2", "King's Quest II: Romancing the Throne"},
|
|
|
|
{"kq3", "King's Quest III: To Heir Is Human"},
|
|
|
|
{"kq4", "King's Quest IV: The Perils of Rosella"},
|
|
|
|
{"lsl1", "Leisure Suit Larry in the Land of the Lounge Lizards"},
|
2007-09-01 14:58:46 +00:00
|
|
|
{"mickey", "Mickey\'s Space Adventure"},
|
2007-01-29 23:25:51 +00:00
|
|
|
{"mixedup", "Mixed-Up Mother Goose"},
|
2007-02-04 13:40:11 +00:00
|
|
|
{"mh1", "Manhunter 1: New York"},
|
|
|
|
{"mh2", "Manhunter 2: San Francisco"},
|
2007-02-09 13:06:57 +00:00
|
|
|
{"pq1", "Police Quest I: In Pursuit of the Death Angel"},
|
2007-01-30 00:22:58 +00:00
|
|
|
{"serguei1", "Serguei's Destiny 1"},
|
|
|
|
{"serguei2", "Serguei's Destiny 2"},
|
2007-02-09 13:06:57 +00:00
|
|
|
{"sq0", "Space Quest 0: Replicated"},
|
2007-02-04 13:40:11 +00:00
|
|
|
{"sq1", "Space Quest I: The Sarien Encounter"},
|
|
|
|
{"sq2", "Space Quest II: Vohaul's Revenge"},
|
2007-02-09 13:06:57 +00:00
|
|
|
{"sqx", "Space Quest X: The Lost Chapter"},
|
2007-01-30 00:22:58 +00:00
|
|
|
{"tetris", "AGI Tetris"},
|
2007-09-18 16:20:44 +00:00
|
|
|
{"troll", "Troll\'s Tale"},
|
2007-09-06 10:48:00 +00:00
|
|
|
{"winnie", "Winnie the Pooh in the Hundred Acre Wood"},
|
2007-01-29 23:25:51 +00:00
|
|
|
{"xmascard", "Xmas Card"},
|
2007-01-29 23:57:55 +00:00
|
|
|
|
2006-12-19 01:11:41 +00:00
|
|
|
{0, 0}
|
|
|
|
};
|
|
|
|
|
2012-03-26 21:21:16 +00:00
|
|
|
static const ExtraGuiOption agiExtraGuiOption = {
|
|
|
|
_s("Use original save/load screens"),
|
|
|
|
_s("Use the original save/load screens, instead of the ScummVM ones"),
|
|
|
|
"originalsaveload",
|
|
|
|
false
|
|
|
|
};
|
|
|
|
|
2010-06-15 10:15:08 +00:00
|
|
|
#include "agi/detection_tables.h"
|
2007-05-26 15:16:21 +00:00
|
|
|
|
2008-03-15 15:25:49 +00:00
|
|
|
using namespace Agi;
|
|
|
|
|
2009-01-29 22:13:01 +00:00
|
|
|
class AgiMetaEngine : public AdvancedMetaEngine {
|
2008-03-15 15:25:49 +00:00
|
|
|
mutable Common::String _gameid;
|
|
|
|
mutable Common::String _extra;
|
|
|
|
|
|
|
|
public:
|
2011-06-10 13:53:50 +00:00
|
|
|
AgiMetaEngine() : AdvancedMetaEngine(Agi::gameDescriptions, sizeof(Agi::AGIGameDescription), agiGames) {
|
2011-06-11 15:52:32 +00:00
|
|
|
_singleid = "agi";
|
2011-10-23 16:52:43 +00:00
|
|
|
_guioptions = GUIO1(GUIO_NOSPEECH);
|
2011-06-10 13:53:50 +00:00
|
|
|
}
|
2008-03-15 15:25:49 +00:00
|
|
|
|
|
|
|
virtual const char *getName() const {
|
2011-05-15 14:50:09 +00:00
|
|
|
return "AGI preAGI + v2 + v3";
|
2008-03-15 15:25:49 +00:00
|
|
|
}
|
2009-03-05 12:04:58 +00:00
|
|
|
virtual const char *getOriginalCopyright() const {
|
2008-03-15 15:25:49 +00:00
|
|
|
return "Sierra AGI Engine (C) Sierra On-Line Software";
|
|
|
|
}
|
|
|
|
|
2008-08-15 18:15:14 +00:00
|
|
|
virtual bool hasFeature(MetaEngineFeature f) const;
|
2009-01-29 22:13:01 +00:00
|
|
|
virtual bool createInstance(OSystem *syst, Engine **engine, const ADGameDescription *desc) const;
|
2012-03-26 21:21:16 +00:00
|
|
|
virtual const ExtraGuiOptions getExtraGuiOptions(const Common::String &target) const;
|
2008-07-28 04:50:27 +00:00
|
|
|
virtual SaveStateList listSaves(const char *target) const;
|
2008-11-09 16:13:34 +00:00
|
|
|
virtual int getMaximumSaveSlot() const;
|
2008-09-11 19:47:45 +00:00
|
|
|
virtual void removeSaveState(const char *target, int slot) const;
|
2008-11-10 19:02:47 +00:00
|
|
|
SaveStateDescriptor querySaveMetaInfos(const char *target, int slot) const;
|
|
|
|
|
2011-06-14 16:02:09 +00:00
|
|
|
const ADGameDescription *fallbackDetect(const FileMap &allFiles, const Common::FSList &fslist) const;
|
2008-03-15 15:25:49 +00:00
|
|
|
};
|
|
|
|
|
2008-08-15 18:15:14 +00:00
|
|
|
bool AgiMetaEngine::hasFeature(MetaEngineFeature f) const {
|
|
|
|
return
|
|
|
|
(f == kSupportsListSaves) ||
|
2008-10-26 16:42:08 +00:00
|
|
|
(f == kSupportsLoadingDuringStartup) ||
|
2008-11-10 19:02:47 +00:00
|
|
|
(f == kSupportsDeleteSave) ||
|
|
|
|
(f == kSavesSupportMetaInfo) ||
|
|
|
|
(f == kSavesSupportThumbnail) ||
|
2012-03-26 21:50:29 +00:00
|
|
|
(f == kSavesSupportCreationDate) ||
|
|
|
|
(f == kSavesSupportPlayTime);
|
2008-08-15 18:15:14 +00:00
|
|
|
}
|
|
|
|
|
2008-11-04 16:11:40 +00:00
|
|
|
bool AgiBase::hasFeature(EngineFeature f) const {
|
2008-11-10 19:02:47 +00:00
|
|
|
return
|
|
|
|
(f == kSupportsRTL) ||
|
|
|
|
(f == kSupportsLoadingDuringRuntime) ||
|
|
|
|
(f == kSupportsSavingDuringRuntime);
|
2008-11-04 16:11:40 +00:00
|
|
|
}
|
|
|
|
|
2008-08-15 18:15:14 +00:00
|
|
|
|
2009-01-29 22:13:01 +00:00
|
|
|
bool AgiMetaEngine::createInstance(OSystem *syst, Engine **engine, const ADGameDescription *desc) const {
|
2008-03-15 15:25:49 +00:00
|
|
|
const Agi::AGIGameDescription *gd = (const Agi::AGIGameDescription *)desc;
|
|
|
|
bool res = true;
|
|
|
|
|
|
|
|
switch (gd->gameType) {
|
|
|
|
case Agi::GType_PreAGI:
|
2011-08-15 15:22:26 +00:00
|
|
|
switch (gd->gameID) {
|
|
|
|
case GID_MICKEY:
|
|
|
|
*engine = new Agi::MickeyEngine(syst, gd);
|
|
|
|
break;
|
2011-08-15 15:33:20 +00:00
|
|
|
case GID_TROLL:
|
|
|
|
*engine = new Agi::TrollEngine(syst, gd);
|
|
|
|
break;
|
2011-08-15 15:22:26 +00:00
|
|
|
case GID_WINNIE:
|
2011-08-15 04:21:19 +00:00
|
|
|
*engine = new Agi::WinnieEngine(syst, gd);
|
2011-08-15 15:22:26 +00:00
|
|
|
break;
|
|
|
|
}
|
2008-03-15 15:25:49 +00:00
|
|
|
break;
|
2011-06-14 12:03:27 +00:00
|
|
|
case Agi::GType_V1:
|
2008-03-15 15:25:49 +00:00
|
|
|
case Agi::GType_V2:
|
|
|
|
case Agi::GType_V3:
|
|
|
|
*engine = new Agi::AgiEngine(syst, gd);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
res = false;
|
|
|
|
error("AGI engine: unknown gameType");
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2012-03-26 21:21:16 +00:00
|
|
|
const ExtraGuiOptions AgiMetaEngine::getExtraGuiOptions(const Common::String &target) const {
|
|
|
|
ExtraGuiOptions options;
|
|
|
|
options.push_back(agiExtraGuiOption);
|
|
|
|
return options;
|
|
|
|
}
|
|
|
|
|
2008-07-28 04:50:27 +00:00
|
|
|
SaveStateList AgiMetaEngine::listSaves(const char *target) const {
|
2011-04-12 14:53:15 +00:00
|
|
|
const uint32 AGIflag = MKTAG('A','G','I',':');
|
2008-07-28 04:50:27 +00:00
|
|
|
Common::SaveFileManager *saveFileMan = g_system->getSavefileManager();
|
2010-03-18 15:54:40 +00:00
|
|
|
Common::StringArray filenames;
|
2008-07-30 21:48:45 +00:00
|
|
|
char saveDesc[31];
|
2008-07-28 04:50:27 +00:00
|
|
|
Common::String pattern = target;
|
|
|
|
pattern += ".???";
|
|
|
|
|
2009-05-29 14:38:22 +00:00
|
|
|
filenames = saveFileMan->listSavefiles(pattern);
|
2008-07-28 04:50:27 +00:00
|
|
|
sort(filenames.begin(), filenames.end()); // Sort (hopefully ensuring we are sorted numerically..)
|
|
|
|
|
|
|
|
SaveStateList saveList;
|
2010-03-18 15:54:40 +00:00
|
|
|
for (Common::StringArray::const_iterator file = filenames.begin(); file != filenames.end(); ++file) {
|
2008-07-28 04:50:27 +00:00
|
|
|
// Obtain the last 3 digits of the filename, since they correspond to the save slot
|
|
|
|
int slotNum = atoi(file->c_str() + file->size() - 3);
|
2009-01-01 15:06:43 +00:00
|
|
|
|
2008-07-28 04:50:27 +00:00
|
|
|
if (slotNum >= 0 && slotNum <= 999) {
|
2009-05-29 14:38:22 +00:00
|
|
|
Common::InSaveFile *in = saveFileMan->openForLoading(*file);
|
2008-07-28 04:50:27 +00:00
|
|
|
if (in) {
|
|
|
|
uint32 type = in->readUint32BE();
|
|
|
|
if (type == AGIflag)
|
|
|
|
in->read(saveDesc, 31);
|
2008-11-05 15:41:12 +00:00
|
|
|
saveList.push_back(SaveStateDescriptor(slotNum, saveDesc));
|
2008-07-28 04:50:27 +00:00
|
|
|
delete in;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return saveList;
|
|
|
|
}
|
|
|
|
|
2008-11-09 16:13:34 +00:00
|
|
|
int AgiMetaEngine::getMaximumSaveSlot() const { return 999; }
|
|
|
|
|
2008-09-11 19:47:45 +00:00
|
|
|
void AgiMetaEngine::removeSaveState(const char *target, int slot) const {
|
2009-06-06 17:44:24 +00:00
|
|
|
char fileName[MAXPATHLEN];
|
2008-11-10 19:02:47 +00:00
|
|
|
sprintf(fileName, "%s.%03d", target, slot);
|
|
|
|
g_system->getSavefileManager()->removeSavefile(fileName);
|
|
|
|
}
|
|
|
|
|
|
|
|
SaveStateDescriptor AgiMetaEngine::querySaveMetaInfos(const char *target, int slot) const {
|
2011-04-12 14:53:15 +00:00
|
|
|
const uint32 AGIflag = MKTAG('A','G','I',':');
|
2009-06-06 17:44:24 +00:00
|
|
|
char fileName[MAXPATHLEN];
|
2008-11-10 19:02:47 +00:00
|
|
|
sprintf(fileName, "%s.%03d", target, slot);
|
|
|
|
|
|
|
|
Common::InSaveFile *in = g_system->getSavefileManager()->openForLoading(fileName);
|
|
|
|
|
|
|
|
if (in) {
|
|
|
|
if (in->readUint32BE() != AGIflag) {
|
|
|
|
delete in;
|
|
|
|
return SaveStateDescriptor();
|
|
|
|
}
|
|
|
|
|
|
|
|
char name[32];
|
|
|
|
in->read(name, 31);
|
|
|
|
|
|
|
|
SaveStateDescriptor desc(slot, name);
|
|
|
|
|
2011-09-25 12:10:43 +00:00
|
|
|
// Do not allow save slot 0 (used for auto-saving) to be deleted or
|
|
|
|
// overwritten.
|
2009-01-27 20:24:08 +00:00
|
|
|
desc.setDeletableFlag(slot != 0);
|
|
|
|
desc.setWriteProtectedFlag(slot == 0);
|
2008-11-10 19:02:47 +00:00
|
|
|
|
|
|
|
char saveVersion = in->readByte();
|
|
|
|
if (saveVersion >= 4) {
|
2011-08-07 12:35:01 +00:00
|
|
|
Graphics::Surface *const thumbnail = Graphics::loadThumbnail(*in);
|
2008-11-10 19:02:47 +00:00
|
|
|
|
|
|
|
desc.setThumbnail(thumbnail);
|
|
|
|
|
|
|
|
uint32 saveDate = in->readUint32BE();
|
|
|
|
uint16 saveTime = in->readUint16BE();
|
2012-03-26 21:50:29 +00:00
|
|
|
if (saveVersion >= 6) {
|
|
|
|
uint32 playTime = in->readUint32BE();
|
|
|
|
desc.setPlayTime(playTime * 1000);
|
|
|
|
}
|
2008-11-10 19:02:47 +00:00
|
|
|
|
|
|
|
int day = (saveDate >> 24) & 0xFF;
|
|
|
|
int month = (saveDate >> 16) & 0xFF;
|
|
|
|
int year = saveDate & 0xFFFF;
|
|
|
|
|
|
|
|
desc.setSaveDate(year, month, day);
|
2009-01-01 15:06:43 +00:00
|
|
|
|
2008-11-10 19:02:47 +00:00
|
|
|
int hour = (saveTime >> 8) & 0xFF;
|
|
|
|
int minutes = saveTime & 0xFF;
|
2008-09-11 19:47:45 +00:00
|
|
|
|
2008-11-10 19:02:47 +00:00
|
|
|
desc.setSaveTime(hour, minutes);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
delete in;
|
|
|
|
|
|
|
|
return desc;
|
2011-09-25 12:01:23 +00:00
|
|
|
} else {
|
2011-09-25 12:10:43 +00:00
|
|
|
SaveStateDescriptor emptySave;
|
|
|
|
// Do not allow save slot 0 (used for auto-saving) to be overwritten.
|
2011-09-25 12:01:23 +00:00
|
|
|
emptySave.setWriteProtectedFlag(slot == 0);
|
|
|
|
return emptySave;
|
2008-11-10 19:02:47 +00:00
|
|
|
}
|
2008-09-11 19:47:45 +00:00
|
|
|
}
|
|
|
|
|
2011-06-14 16:02:09 +00:00
|
|
|
const ADGameDescription *AgiMetaEngine::fallbackDetect(const FileMap &allFilesXXX, const Common::FSList &fslist) const {
|
2008-06-11 06:00:56 +00:00
|
|
|
typedef Common::HashMap<Common::String, int32> IntMap;
|
2007-05-26 15:16:21 +00:00
|
|
|
IntMap allFiles;
|
2007-06-12 12:22:25 +00:00
|
|
|
bool matchedUsingFilenames = false;
|
2007-06-14 12:06:12 +00:00
|
|
|
bool matchedUsingWag = false;
|
|
|
|
int wagFileCount = 0;
|
|
|
|
WagFileParser wagFileParser;
|
2008-10-02 16:58:59 +00:00
|
|
|
Common::FSNode wagFileNode;
|
2008-03-15 15:25:49 +00:00
|
|
|
Common::String description;
|
2007-06-12 12:22:25 +00:00
|
|
|
|
2008-03-15 15:25:49 +00:00
|
|
|
// // Set the defaults for gameid and extra
|
|
|
|
_gameid = "agi-fanmade";
|
|
|
|
_extra.clear();
|
|
|
|
|
2007-06-12 12:22:25 +00:00
|
|
|
// Set the default values for the fallback descriptor's ADGameDescription part.
|
|
|
|
g_fallbackDesc.desc.language = Common::UNK_LANG;
|
2013-05-02 22:26:58 +00:00
|
|
|
g_fallbackDesc.desc.platform = Common::kPlatformDOS;
|
2009-01-29 22:13:01 +00:00
|
|
|
g_fallbackDesc.desc.flags = ADGF_NO_FLAGS;
|
2007-05-26 15:16:21 +00:00
|
|
|
|
2007-06-12 12:22:25 +00:00
|
|
|
// Set default values for the fallback descriptor's AGIGameDescription part.
|
|
|
|
g_fallbackDesc.gameID = GID_FANMADE;
|
|
|
|
g_fallbackDesc.features = GF_FANMADE;
|
|
|
|
g_fallbackDesc.version = 0x2917;
|
2007-05-26 15:16:21 +00:00
|
|
|
|
2007-06-14 12:06:12 +00:00
|
|
|
// First grab all filenames and at the same time count the number of *.wag files
|
2008-11-06 13:59:39 +00:00
|
|
|
for (Common::FSList::const_iterator file = fslist.begin(); file != fslist.end(); ++file) {
|
2007-05-26 15:16:21 +00:00
|
|
|
if (file->isDirectory()) continue;
|
2007-06-23 18:51:33 +00:00
|
|
|
Common::String filename = file->getName();
|
2007-06-12 12:22:25 +00:00
|
|
|
filename.toLowercase();
|
2011-06-04 16:16:50 +00:00
|
|
|
allFiles[filename] = true; // Save the filename in a hash table
|
2007-09-19 08:40:12 +00:00
|
|
|
|
2007-06-14 12:06:12 +00:00
|
|
|
if (filename.hasSuffix(".wag")) {
|
|
|
|
// Save latest found *.wag file's path (Can be used to open the file, the name can't)
|
2008-09-30 16:38:46 +00:00
|
|
|
wagFileNode = *file;
|
2007-06-14 12:06:12 +00:00
|
|
|
wagFileCount++; // Count found *.wag files
|
|
|
|
}
|
2007-05-26 15:16:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (allFiles.contains("logdir") && allFiles.contains("object") &&
|
|
|
|
allFiles.contains("picdir") && allFiles.contains("snddir") &&
|
|
|
|
allFiles.contains("viewdir") && allFiles.contains("vol.0") &&
|
2007-06-12 12:22:25 +00:00
|
|
|
allFiles.contains("words.tok")) { // Check for v2
|
|
|
|
|
|
|
|
// The default AGI interpreter version 0x2917 is okay for v2 games
|
|
|
|
// so we don't have to change it here.
|
|
|
|
matchedUsingFilenames = true;
|
|
|
|
|
2008-02-12 15:20:47 +00:00
|
|
|
// Check for AGIPAL by checking for existence of any of the files "pal.100" - "pal.109"
|
|
|
|
bool agipal = false;
|
|
|
|
char agipalFile[] = "pal.xxx";
|
|
|
|
for (uint i = 100; i <= 109; i++) {
|
|
|
|
sprintf(agipalFile, "pal.%d", i);
|
|
|
|
if (allFiles.contains(agipalFile)) {
|
|
|
|
agipal = true; // We found a file "pal.x" where 100 <= x <= 109 so it's AGIPAL
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (agipal) { // Check if it is AGIPAL
|
2007-06-12 12:22:25 +00:00
|
|
|
description = "Unknown v2 AGIPAL Game";
|
|
|
|
g_fallbackDesc.features |= GF_AGIPAL; // Add AGIPAL feature flag
|
|
|
|
} else { // Not AGIPAL so just plain v2
|
|
|
|
description = "Unknown v2 Game";
|
2007-09-19 08:40:12 +00:00
|
|
|
}
|
2007-05-26 15:16:21 +00:00
|
|
|
} else { // Try v3
|
|
|
|
char name[8];
|
|
|
|
|
|
|
|
for (IntMap::const_iterator f = allFiles.begin(); f != allFiles.end(); ++f) {
|
|
|
|
if (f->_key.hasSuffix("vol.0")) {
|
|
|
|
memset(name, 0, 8);
|
2007-06-06 13:35:41 +00:00
|
|
|
strncpy(name, f->_key.c_str(), MIN((uint)8, f->_key.size() > 5 ? f->_key.size() - 5 : f->_key.size()));
|
2007-05-26 15:16:21 +00:00
|
|
|
|
|
|
|
if (allFiles.contains("object") && allFiles.contains("words.tok") &&
|
|
|
|
allFiles.contains(Common::String(name) + "dir")) {
|
2007-06-12 12:22:25 +00:00
|
|
|
matchedUsingFilenames = true;
|
|
|
|
description = "Unknown v3 Game";
|
|
|
|
g_fallbackDesc.version = 0x3149; // Set the default AGI version for an AGI v3 game
|
2007-05-26 15:16:21 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2007-06-14 12:06:12 +00:00
|
|
|
|
|
|
|
// WinAGI produces *.wag file with interpreter version, game name and other parameters.
|
|
|
|
// If there's exactly one *.wag file and it parses successfully then we'll use its information.
|
2008-09-30 16:38:46 +00:00
|
|
|
if (wagFileCount == 1 && wagFileParser.parse(wagFileNode)) {
|
2007-06-14 12:06:12 +00:00
|
|
|
matchedUsingWag = true;
|
|
|
|
|
|
|
|
const WagProperty *wagAgiVer = wagFileParser.getProperty(WagProperty::PC_INTVERSION);
|
|
|
|
const WagProperty *wagGameID = wagFileParser.getProperty(WagProperty::PC_GAMEID);
|
|
|
|
const WagProperty *wagGameDesc = wagFileParser.getProperty(WagProperty::PC_GAMEDESC);
|
|
|
|
const WagProperty *wagGameVer = wagFileParser.getProperty(WagProperty::PC_GAMEVERSION);
|
|
|
|
const WagProperty *wagGameLastEdit = wagFileParser.getProperty(WagProperty::PC_GAMELAST);
|
|
|
|
|
|
|
|
// If there is an AGI version number in the *.wag file then let's use it
|
|
|
|
if (wagAgiVer != NULL && wagFileParser.checkAgiVersionProperty(*wagAgiVer)) {
|
|
|
|
// TODO/FIXME: Check that version number is something we support before trying to use it.
|
|
|
|
// If the version number is unsupported then it'll get switched to 0x2917 later.
|
|
|
|
// But there's the possibility that file based detection has detected something else
|
|
|
|
// than a v2 AGI game. So there's a possibility for conflicting information.
|
|
|
|
g_fallbackDesc.version = wagFileParser.convertToAgiVersionNumber(*wagAgiVer);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set gameid according to *.wag file information if it's present and it doesn't contain whitespace.
|
|
|
|
if (wagGameID != NULL && !Common::String(wagGameID->getData()).contains(" ")) {
|
2008-03-15 15:25:49 +00:00
|
|
|
_gameid = wagGameID->getData();
|
|
|
|
debug(3, "Agi::fallbackDetector: Using game id (%s) from WAG file", _gameid.c_str());
|
2007-06-14 12:06:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Set game description and extra according to *.wag file information if they're present
|
|
|
|
if (wagGameDesc != NULL) {
|
|
|
|
description = wagGameDesc->getData();
|
|
|
|
debug(3, "Agi::fallbackDetector: Game description (%s) from WAG file", wagGameDesc->getData());
|
|
|
|
|
|
|
|
// If there's game version in the *.wag file, set extra to it
|
|
|
|
if (wagGameVer != NULL) {
|
2008-03-15 15:25:49 +00:00
|
|
|
_extra = wagGameVer->getData();
|
2007-06-14 12:06:12 +00:00
|
|
|
debug(3, "Agi::fallbackDetector: Game version (%s) from WAG file", wagGameVer->getData());
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there's game last edit date in the *.wag file, add it to extra
|
|
|
|
if (wagGameLastEdit != NULL) {
|
2010-10-12 04:19:58 +00:00
|
|
|
if (!_extra.empty())
|
|
|
|
_extra += " ";
|
2008-03-15 15:25:49 +00:00
|
|
|
_extra += wagGameLastEdit->getData();
|
2007-06-14 12:06:12 +00:00
|
|
|
debug(3, "Agi::fallbackDetector: Game's last edit date (%s) from WAG file", wagGameLastEdit->getData());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (wagFileCount > 1) { // More than one *.wag file, confusing! So let's not use them.
|
|
|
|
warning("More than one (%d) *.wag files found. WAG files ignored", wagFileCount);
|
|
|
|
}
|
|
|
|
|
2007-06-12 12:22:25 +00:00
|
|
|
// Check that the AGI interpreter version is a supported one
|
|
|
|
if (!(g_fallbackDesc.version >= 0x2000 && g_fallbackDesc.version < 0x4000)) {
|
|
|
|
warning("Unsupported AGI interpreter version 0x%x in AGI's fallback detection. Using default 0x2917", g_fallbackDesc.version);
|
|
|
|
g_fallbackDesc.version = 0x2917;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set game type (v2 or v3) according to the AGI interpreter version number
|
|
|
|
if (g_fallbackDesc.version >= 0x2000 && g_fallbackDesc.version < 0x3000)
|
|
|
|
g_fallbackDesc.gameType = GType_V2;
|
|
|
|
else if (g_fallbackDesc.version >= 0x3000 && g_fallbackDesc.version < 0x4000)
|
|
|
|
g_fallbackDesc.gameType = GType_V3;
|
|
|
|
|
|
|
|
// Check if we found a match with any of the fallback methods
|
2007-06-14 12:06:12 +00:00
|
|
|
if (matchedUsingWag || matchedUsingFilenames) {
|
2008-03-15 15:25:49 +00:00
|
|
|
_extra = description + (!_extra.empty() ? " " : "") + _extra; // Let's combine the description and extra
|
2009-01-01 15:06:43 +00:00
|
|
|
|
|
|
|
// Override the gameid & extra values in g_fallbackDesc.desc. This only works
|
2008-03-15 15:25:49 +00:00
|
|
|
// until the fallback detector is called again, and while the MetaEngine instance
|
|
|
|
// is alive (as else the string storage is modified/deleted).
|
|
|
|
g_fallbackDesc.desc.gameid = _gameid.c_str();
|
|
|
|
g_fallbackDesc.desc.extra = _extra.c_str();
|
2007-05-26 15:16:21 +00:00
|
|
|
|
2012-08-15 07:51:55 +00:00
|
|
|
Common::String fallbackWarning;
|
|
|
|
|
|
|
|
fallbackWarning = "Your game version has been detected using fallback matching as a\n";
|
|
|
|
fallbackWarning += Common::String::format("variant of %s (%s).\n", g_fallbackDesc.desc.gameid, g_fallbackDesc.desc.extra);
|
|
|
|
fallbackWarning += "If this is an original and unmodified version or new made Fanmade game,\n";
|
|
|
|
fallbackWarning += "please report any, information previously printed by ScummVM to the team.\n";
|
|
|
|
|
|
|
|
g_system->logMessage(LogMessageType::kWarning, fallbackWarning.c_str());
|
2007-06-12 12:22:25 +00:00
|
|
|
|
2009-01-29 22:13:01 +00:00
|
|
|
return (const ADGameDescription *)&g_fallbackDesc;
|
2007-05-26 15:16:21 +00:00
|
|
|
}
|
|
|
|
|
2008-03-15 15:25:49 +00:00
|
|
|
return 0;
|
2007-09-01 14:58:46 +00:00
|
|
|
}
|
2007-01-24 22:42:44 +00:00
|
|
|
|
2008-05-06 03:00:26 +00:00
|
|
|
#if PLUGIN_ENABLED_DYNAMIC(AGI)
|
|
|
|
REGISTER_PLUGIN_DYNAMIC(AGI, PLUGIN_TYPE_ENGINE, AgiMetaEngine);
|
|
|
|
#else
|
|
|
|
REGISTER_PLUGIN_STATIC(AGI, PLUGIN_TYPE_ENGINE, AgiMetaEngine);
|
|
|
|
#endif
|
2008-11-10 19:02:47 +00:00
|
|
|
|
|
|
|
namespace Agi {
|
|
|
|
|
2009-01-01 15:06:43 +00:00
|
|
|
bool AgiBase::canLoadGameStateCurrently() {
|
2009-06-06 17:43:51 +00:00
|
|
|
return (!(getGameType() == GType_PreAGI) && getflag(fMenusWork) && !_noSaveLoadAllowed);
|
2008-11-10 19:02:47 +00:00
|
|
|
}
|
|
|
|
|
2009-01-01 15:06:43 +00:00
|
|
|
bool AgiBase::canSaveGameStateCurrently() {
|
2010-06-15 10:29:43 +00:00
|
|
|
if (getGameID() == GID_BC) // Technically in Black Cauldron we may save anytime
|
|
|
|
return true;
|
2011-06-19 22:59:48 +00:00
|
|
|
|
2009-06-06 17:42:37 +00:00
|
|
|
return (!(getGameType() == GType_PreAGI) && getflag(fMenusWork) && !_noSaveLoadAllowed && _game.inputEnabled);
|
2008-11-10 19:02:47 +00:00
|
|
|
}
|
|
|
|
|
2009-07-22 15:55:33 +00:00
|
|
|
int AgiEngine::agiDetectGame() {
|
|
|
|
int ec = errOK;
|
|
|
|
|
|
|
|
assert(_gameDescription != NULL);
|
|
|
|
|
2011-05-25 13:27:35 +00:00
|
|
|
if (getVersion() <= 0x2001) {
|
2011-06-04 15:12:26 +00:00
|
|
|
_loader = new AgiLoader_v1(this);
|
2011-05-25 13:27:35 +00:00
|
|
|
} else if (getVersion() <= 0x2999) {
|
2009-07-22 15:55:33 +00:00
|
|
|
_loader = new AgiLoader_v2(this);
|
|
|
|
} else {
|
|
|
|
_loader = new AgiLoader_v3(this);
|
|
|
|
}
|
|
|
|
ec = _loader->detectGame();
|
|
|
|
|
|
|
|
return ec;
|
|
|
|
}
|
|
|
|
|
2008-11-10 19:02:47 +00:00
|
|
|
} // End of namespace Agi
|