2018-10-14 13:02:24 -07:00
|
|
|
/* ScummVM - Graphic Adventure Engine
|
|
|
|
*
|
|
|
|
* ScummVM is the legal property of its developers, whose names
|
|
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
|
|
* file distributed with this source distribution.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "gargoyle/gargoyle.h"
|
|
|
|
|
|
|
|
#include "base/plugins.h"
|
2018-11-03 20:41:59 -07:00
|
|
|
#include "common/md5.h"
|
|
|
|
#include "common/memstream.h"
|
2018-10-14 13:02:24 -07:00
|
|
|
#include "common/savefile.h"
|
|
|
|
#include "common/str-array.h"
|
|
|
|
#include "common/system.h"
|
2018-11-03 20:41:59 -07:00
|
|
|
#include "engines/advancedDetector.h"
|
2018-10-14 13:02:24 -07:00
|
|
|
#include "graphics/colormasks.h"
|
|
|
|
#include "graphics/surface.h"
|
|
|
|
|
|
|
|
#define MAX_SAVES 99
|
|
|
|
|
|
|
|
namespace Gargoyle {
|
|
|
|
|
|
|
|
struct GargoyleGameDescription {
|
2018-11-03 20:41:59 -07:00
|
|
|
ADGameDescription _desc;
|
|
|
|
Common::String _filename;
|
|
|
|
InterpreterType _interpType;
|
|
|
|
Common::String _md5;
|
2018-10-14 13:02:24 -07:00
|
|
|
};
|
|
|
|
|
2018-10-17 22:45:25 -07:00
|
|
|
const Common::String &GargoyleEngine::getFilename() const {
|
2018-11-03 20:41:59 -07:00
|
|
|
return _gameDescription->_filename;
|
2018-10-17 22:45:25 -07:00
|
|
|
}
|
2018-10-14 13:02:24 -07:00
|
|
|
uint32 GargoyleEngine::getFeatures() const {
|
2018-11-03 20:41:59 -07:00
|
|
|
return _gameDescription->_desc.flags;
|
2018-10-14 13:02:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
bool GargoyleEngine::isDemo() const {
|
2018-11-03 20:41:59 -07:00
|
|
|
return (bool)(_gameDescription->_desc.flags & ADGF_DEMO);
|
2018-10-14 13:02:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
Common::Language GargoyleEngine::getLanguage() const {
|
2018-11-03 20:41:59 -07:00
|
|
|
return _gameDescription->_desc.language;
|
2018-10-14 13:02:24 -07:00
|
|
|
}
|
|
|
|
|
2018-10-16 21:06:07 -07:00
|
|
|
InterpreterType GargoyleEngine::getInterpreterType() const {
|
2018-11-03 20:41:59 -07:00
|
|
|
return _gameDescription->_interpType;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Common::String &GargoyleEngine::getGameMD5() const {
|
|
|
|
return _gameDescription->_md5;
|
2018-10-16 21:06:07 -07:00
|
|
|
}
|
|
|
|
|
2018-10-14 13:02:24 -07:00
|
|
|
} // End of namespace Gargoyle
|
|
|
|
|
2018-10-16 22:29:06 -07:00
|
|
|
static const PlainGameDescriptor gargoyleGames[] = {
|
2018-11-10 12:21:23 -08:00
|
|
|
{"zcode", "Zcode Games" },
|
2018-11-03 21:18:23 -07:00
|
|
|
{"scottadams", "Scott Adams Games"},
|
|
|
|
|
|
|
|
// Scott Adams games
|
|
|
|
{ "adventureland", "Adventureland" },
|
|
|
|
{ "pirateadventure", "Pirate Adventure" },
|
|
|
|
{ "missionimpossible", "Mission Impossible" },
|
|
|
|
{ "voodoocastle", "Voodoo Castle" },
|
|
|
|
{ "thecount", "The Count" },
|
|
|
|
{ "strangeodyssey", "Strange Odyssey" },
|
|
|
|
{ "mysteryfunhouse", "Mystery Fun House" },
|
|
|
|
{ "pyramidofdoom", "Pyramid Of Doom" },
|
|
|
|
{ "ghosttown", "Ghost Town" },
|
|
|
|
{ "savageisland1", "Savage Island, Part 1" },
|
|
|
|
{ "savageisland2", "Savage Island, Part 2" },
|
|
|
|
{ "goldenvoyage", "The Golden Voyage" },
|
|
|
|
{ "adventure13", "Adventure 13" },
|
|
|
|
{ "adventure14", "Adventure 14" },
|
|
|
|
{ "buckaroobonzai", "Buckaroo Banzai" },
|
|
|
|
|
2018-10-14 13:02:24 -07:00
|
|
|
{0, 0}
|
|
|
|
};
|
|
|
|
|
2018-10-17 22:45:25 -07:00
|
|
|
#include "common/config-manager.h"
|
2018-10-18 07:34:39 -07:00
|
|
|
#include "common/file.h"
|
2018-10-14 13:02:24 -07:00
|
|
|
#include "gargoyle/detection_tables.h"
|
2018-11-10 12:21:23 -08:00
|
|
|
#include "gargoyle/frotz/detection.h"
|
|
|
|
#include "gargoyle/frotz/frotz.h"
|
2018-10-17 21:14:04 -07:00
|
|
|
#include "gargoyle/scott/detection.h"
|
2018-10-16 22:29:06 -07:00
|
|
|
#include "gargoyle/scott/scott.h"
|
2018-10-14 13:02:24 -07:00
|
|
|
|
|
|
|
class GargoyleMetaEngine : public AdvancedMetaEngine {
|
|
|
|
public:
|
2018-10-16 22:29:06 -07:00
|
|
|
GargoyleMetaEngine() : AdvancedMetaEngine(Gargoyle::gameDescriptions, sizeof(Gargoyle::GargoyleGameDescription), gargoyleGames) {
|
2018-10-14 13:02:24 -07:00
|
|
|
_maxScanDepth = 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual const char *getName() const {
|
|
|
|
return "Gargoyle Engine";
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual const char *getOriginalCopyright() const {
|
2018-11-03 21:18:23 -07:00
|
|
|
return "Gargoyle Engine (c) 2018";
|
2018-10-14 13:02:24 -07:00
|
|
|
}
|
|
|
|
|
2018-10-17 21:14:04 -07:00
|
|
|
virtual bool hasFeature(MetaEngineFeature f) const override;
|
|
|
|
virtual bool createInstance(OSystem *syst, Engine **engine, const ADGameDescription *desc) const override;
|
2018-10-14 13:02:24 -07:00
|
|
|
virtual SaveStateList listSaves(const char *target) const;
|
|
|
|
virtual int getMaximumSaveSlot() const;
|
|
|
|
virtual void removeSaveState(const char *target, int slot) const;
|
|
|
|
SaveStateDescriptor querySaveMetaInfos(const char *target, int slot) const;
|
2018-10-17 21:14:04 -07:00
|
|
|
|
|
|
|
virtual DetectedGames detectGames(const Common::FSList &fslist) const override;
|
2018-10-17 22:45:25 -07:00
|
|
|
|
|
|
|
virtual ADDetectedGames detectGame(const Common::FSNode &parent, const FileMap &allFiles, Common::Language language, Common::Platform platform, const Common::String &extra) const override;
|
2018-10-14 13:02:24 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
bool GargoyleMetaEngine::hasFeature(MetaEngineFeature f) const {
|
|
|
|
return
|
2018-11-10 05:42:22 +00:00
|
|
|
(f == kSupportsListSaves) ||
|
|
|
|
(f == kSupportsLoadingDuringStartup) ||
|
|
|
|
(f == kSupportsDeleteSave) ||
|
|
|
|
(f == kSavesSupportMetaInfo) ||
|
|
|
|
(f == kSavesSupportCreationDate) ||
|
|
|
|
(f == kSavesSupportPlayTime) ||
|
|
|
|
(f == kSimpleSavesNames);
|
2018-10-14 13:02:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Gargoyle::GargoyleEngine::hasFeature(EngineFeature f) const {
|
|
|
|
return
|
2018-11-10 05:42:22 +00:00
|
|
|
(f == kSupportsRTL) ||
|
|
|
|
(f == kSupportsLoadingDuringRuntime) ||
|
|
|
|
(f == kSupportsSavingDuringRuntime);
|
2018-10-14 13:02:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
bool GargoyleMetaEngine::createInstance(OSystem *syst, Engine **engine, const ADGameDescription *desc) const {
|
2018-11-10 18:25:20 +00:00
|
|
|
const Gargoyle::GargoyleGameDescription *gd = (const Gargoyle::GargoyleGameDescription *)desc;
|
2018-10-17 22:45:25 -07:00
|
|
|
|
2018-11-03 20:41:59 -07:00
|
|
|
switch (gd->_interpType) {
|
2018-11-10 12:21:23 -08:00
|
|
|
case Gargoyle::INTERPRETER_FROTZ:
|
|
|
|
*engine = new Gargoyle::Frotz::Frotz(syst, gd);
|
|
|
|
break;
|
2018-10-16 22:29:06 -07:00
|
|
|
case Gargoyle::INTERPRETER_SCOTT:
|
|
|
|
*engine = new Gargoyle::Scott::Scott(syst, gd);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
error("Unknown interpreter");
|
|
|
|
}
|
2018-10-14 13:02:24 -07:00
|
|
|
|
|
|
|
return gd != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
SaveStateList GargoyleMetaEngine::listSaves(const char *target) const {
|
2018-11-04 15:11:11 -08:00
|
|
|
Common::SaveFileManager *saveFileMan = g_system->getSavefileManager();
|
|
|
|
Common::StringArray filenames;
|
|
|
|
Common::String saveDesc;
|
|
|
|
Common::String pattern = Common::String::format("%s.0##", target);
|
|
|
|
Gargoyle::SavegameHeader header;
|
|
|
|
|
|
|
|
filenames = saveFileMan->listSavefiles(pattern);
|
|
|
|
|
2018-10-14 13:02:24 -07:00
|
|
|
SaveStateList saveList;
|
2018-11-04 15:11:11 -08:00
|
|
|
for (Common::StringArray::const_iterator file = filenames.begin(); file != filenames.end(); ++file) {
|
|
|
|
const char *ext = strrchr(file->c_str(), '.');
|
|
|
|
int slot = ext ? atoi(ext + 1) : -1;
|
|
|
|
|
|
|
|
if (slot >= 0 && slot <= MAX_SAVES) {
|
|
|
|
Common::InSaveFile *in = g_system->getSavefileManager()->openForLoading(*file);
|
|
|
|
|
|
|
|
if (in) {
|
|
|
|
if (Gargoyle::FileStream::readSavegameHeader(in, header))
|
|
|
|
saveList.push_back(SaveStateDescriptor(slot, header._saveName));
|
|
|
|
|
|
|
|
delete in;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sort saves based on slot number.
|
|
|
|
Common::sort(saveList.begin(), saveList.end(), SaveStateDescriptorSlotComparator());
|
2018-10-14 13:02:24 -07:00
|
|
|
return saveList;
|
|
|
|
}
|
|
|
|
|
|
|
|
int GargoyleMetaEngine::getMaximumSaveSlot() const {
|
|
|
|
return MAX_SAVES;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GargoyleMetaEngine::removeSaveState(const char *target, int slot) const {
|
|
|
|
}
|
|
|
|
|
|
|
|
SaveStateDescriptor GargoyleMetaEngine::querySaveMetaInfos(const char *target, int slot) const {
|
2018-11-04 18:02:36 -08:00
|
|
|
Common::String filename = Common::String::format("%s.%03d", target, slot);
|
|
|
|
Common::InSaveFile *in = g_system->getSavefileManager()->openForLoading(filename);
|
|
|
|
|
|
|
|
if (in) {
|
|
|
|
Gargoyle::SavegameHeader header;
|
|
|
|
if (Gargoyle::FileStream::readSavegameHeader(in, header)) {
|
|
|
|
// Create the return descriptor
|
|
|
|
SaveStateDescriptor desc(slot, header._saveName);
|
|
|
|
desc.setSaveDate(header._year, header._month, header._day);
|
|
|
|
desc.setSaveTime(header._hour, header._minute);
|
|
|
|
desc.setPlayTime(header._totalFrames * GAME_FRAME_TIME);
|
|
|
|
|
|
|
|
delete in;
|
|
|
|
return desc;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-14 13:02:24 -07:00
|
|
|
return SaveStateDescriptor();
|
|
|
|
}
|
|
|
|
|
2018-10-17 21:14:04 -07:00
|
|
|
DetectedGames GargoyleMetaEngine::detectGames(const Common::FSList &fslist) const {
|
|
|
|
DetectedGames detectedGames;
|
2018-11-10 12:21:23 -08:00
|
|
|
Gargoyle::Frotz::FrotzMetaEngine::detectGames(fslist, detectedGames);
|
2018-10-17 21:14:04 -07:00
|
|
|
Gargoyle::Scott::ScottMetaEngine::detectGames(fslist, detectedGames);
|
|
|
|
|
|
|
|
return detectedGames;
|
|
|
|
}
|
2018-10-14 13:02:24 -07:00
|
|
|
|
2018-10-17 22:45:25 -07:00
|
|
|
static Gargoyle::GargoyleGameDescription gameDescription;
|
|
|
|
|
|
|
|
ADDetectedGames GargoyleMetaEngine::detectGame(const Common::FSNode &parent, const FileMap &allFiles, Common::Language language, Common::Platform platform, const Common::String &extra) const {
|
|
|
|
static char gameId[100];
|
|
|
|
strcpy(gameId, ConfMan.get("gameid").c_str());
|
2018-10-18 07:34:39 -07:00
|
|
|
Common::String filename = ConfMan.get("filename");
|
2018-11-03 21:05:59 -07:00
|
|
|
|
|
|
|
Common::FSList fslist;
|
|
|
|
DetectedGames detectedGames;
|
|
|
|
fslist.push_back(parent.getChild(filename));
|
|
|
|
ADDetectedGames results;
|
2018-11-03 20:41:59 -07:00
|
|
|
Common::File f;
|
|
|
|
|
2018-11-10 12:21:23 -08:00
|
|
|
// Check each sub-engine for any detected games
|
|
|
|
if (Gargoyle::Frotz::FrotzMetaEngine::detectGames(fslist, detectedGames))
|
|
|
|
gameDescription._interpType = Gargoyle::INTERPRETER_FROTZ;
|
|
|
|
else if (Gargoyle::Scott::ScottMetaEngine::detectGames(fslist, detectedGames))
|
|
|
|
gameDescription._interpType = Gargoyle::INTERPRETER_SCOTT;
|
|
|
|
else
|
|
|
|
// No match found, so return no results
|
|
|
|
return results;
|
|
|
|
|
|
|
|
// Set up the game description and return it
|
|
|
|
if (f.open(parent.getChild(filename))) {
|
2018-11-03 21:05:59 -07:00
|
|
|
DetectedGame gd = detectedGames.front();
|
|
|
|
|
2018-11-03 20:41:59 -07:00
|
|
|
gameDescription._desc.gameId = gameId;
|
2018-11-03 21:05:59 -07:00
|
|
|
gameDescription._desc.language = gd.language;
|
|
|
|
gameDescription._desc.platform = gd.platform;
|
2018-11-09 20:12:32 -08:00
|
|
|
gameDescription._desc.guiOptions = GUIO4(GUIO_NOSPEECH, GUIO_NOSFX, GUIO_NOMUSIC, GUIO_NOSUBTITLES);
|
2018-11-03 20:41:59 -07:00
|
|
|
gameDescription._filename = filename;
|
|
|
|
gameDescription._md5 = Common::computeStreamMD5AsString(f, 5000);
|
2018-10-18 07:34:39 -07:00
|
|
|
|
|
|
|
ADDetectedGame dg((ADGameDescription *)&gameDescription);
|
2018-11-03 21:05:59 -07:00
|
|
|
results.push_back(dg);
|
2018-10-18 07:34:39 -07:00
|
|
|
}
|
2018-10-17 22:45:25 -07:00
|
|
|
|
2018-11-03 21:05:59 -07:00
|
|
|
return results;
|
2018-10-17 22:45:25 -07:00
|
|
|
}
|
|
|
|
|
2018-10-14 13:02:24 -07:00
|
|
|
#if PLUGIN_ENABLED_DYNAMIC(GARGOYLE)
|
2018-11-10 05:42:22 +00:00
|
|
|
REGISTER_PLUGIN_DYNAMIC(Gargoyle, PLUGIN_TYPE_ENGINE, GargoyleMetaEngine);
|
2018-10-14 13:02:24 -07:00
|
|
|
#else
|
2018-11-10 05:42:22 +00:00
|
|
|
REGISTER_PLUGIN_STATIC(GARGOYLE, PLUGIN_TYPE_ENGINE, GargoyleMetaEngine);
|
2018-10-14 13:02:24 -07:00
|
|
|
#endif
|