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-05-03 14:24:48 +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.
|
2011-12-11 01:57:03 +01:00
|
|
|
*
|
2006-05-03 14:24:48 +00:00
|
|
|
*/
|
|
|
|
|
2008-05-11 22:32:37 +00:00
|
|
|
#include "kyra/kyra_lok.h"
|
2009-03-04 11:53:25 +00:00
|
|
|
#include "kyra/lol.h"
|
2008-05-02 14:46:30 +00:00
|
|
|
#include "kyra/kyra_hof.h"
|
|
|
|
#include "kyra/kyra_mr.h"
|
2011-12-12 16:14:08 +01:00
|
|
|
#include "kyra/eob.h"
|
|
|
|
#include "kyra/darkmoon.h"
|
2009-03-04 00:45:34 +00:00
|
|
|
|
2006-05-03 14:24:48 +00:00
|
|
|
#include "common/config-manager.h"
|
2011-04-24 11:34:27 +03:00
|
|
|
#include "common/system.h"
|
2008-03-27 18:03:00 +00:00
|
|
|
#include "common/savefile.h"
|
2012-02-29 17:06:13 +01:00
|
|
|
#include "common/translation.h"
|
2006-05-03 14:24:48 +00:00
|
|
|
|
2011-04-24 11:34:27 +03:00
|
|
|
#include "engines/advancedDetector.h"
|
|
|
|
|
2006-05-03 14:24:48 +00:00
|
|
|
#include "base/plugins.h"
|
|
|
|
|
2006-11-09 14:38:16 +00:00
|
|
|
struct KYRAGameDescription {
|
2009-01-29 22:13:01 +00:00
|
|
|
ADGameDescription desc;
|
2006-11-09 14:38:16 +00:00
|
|
|
|
2007-01-25 21:16:57 +00:00
|
|
|
Kyra::GameFlags flags;
|
2006-05-03 14:24:48 +00:00
|
|
|
};
|
|
|
|
|
2010-06-15 10:15:51 +00:00
|
|
|
#include "kyra/detection_tables.h"
|
2006-11-09 14:38:16 +00:00
|
|
|
|
2010-08-24 13:11:12 +00:00
|
|
|
namespace {
|
|
|
|
|
2011-12-26 21:19:29 +01:00
|
|
|
const char *const directoryGlobs[] = {
|
2010-08-24 13:11:12 +00:00
|
|
|
"malcolm",
|
|
|
|
0
|
|
|
|
};
|
|
|
|
|
2012-02-29 17:06:13 +01:00
|
|
|
const ADExtraGuiOptionsMap gameGuiOptions[] = {
|
|
|
|
// Kyrandia 3 options
|
|
|
|
|
|
|
|
{
|
|
|
|
GAMEOPTION_KYRA3_AUDIENCE,
|
|
|
|
{
|
2012-03-25 11:47:20 +03:00
|
|
|
// I18N: Studio audience adds an applause and cheering sounds whenever
|
2012-03-25 17:05:23 +03:00
|
|
|
// Malcolm makes a joke.
|
2012-02-29 17:06:13 +01:00
|
|
|
_s("Studio audience"),
|
|
|
|
_s("Enable studio audience"),
|
|
|
|
"studio_audience",
|
|
|
|
true
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
GAMEOPTION_KYRA3_SKIP,
|
|
|
|
{
|
2012-03-25 11:47:20 +03:00
|
|
|
// I18N: This option allows the user to skip text and cutscenes.
|
2012-02-29 17:06:13 +01:00
|
|
|
_s("Skip support"),
|
|
|
|
_s("Allow text and cutscenes to be skipped"),
|
|
|
|
"skip_support",
|
|
|
|
true
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
GAMEOPTION_KYRA3_HELIUM,
|
|
|
|
{
|
2012-03-25 11:47:20 +03:00
|
|
|
// I18N: Helium mode makes people sound like they've inhaled Helium.
|
2012-02-29 17:06:13 +01:00
|
|
|
_s("Helium mode"),
|
|
|
|
_s("Enable helium mode"),
|
|
|
|
"helium_mode",
|
|
|
|
false
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
#ifdef ENABLE_LOL
|
|
|
|
// LoL options
|
|
|
|
|
|
|
|
{
|
|
|
|
GAMEOPTION_LOL_SCROLLING,
|
|
|
|
{
|
2012-03-25 17:05:23 +03:00
|
|
|
// I18N: When enabled, this option makes scrolling smoother when
|
|
|
|
// changing from one screen to another.
|
2012-02-29 17:06:13 +01:00
|
|
|
_s("Smooth scrolling"),
|
|
|
|
_s("Enable smooth scrolling when walking"),
|
|
|
|
"smooth_scrolling",
|
|
|
|
true
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
{
|
|
|
|
GAMEOPTION_LOL_CURSORS,
|
|
|
|
{
|
2012-03-25 11:47:20 +03:00
|
|
|
// I18N: When enabled, this option changes the cursor when it floats to the
|
|
|
|
// edge of the screen to a directional arrow. The player can then click to
|
|
|
|
// walk towards that direction.
|
2012-02-29 17:06:13 +01:00
|
|
|
_s("Floating cursors"),
|
|
|
|
_s("Enable floating cursors"),
|
|
|
|
"floating_cursors",
|
|
|
|
false
|
|
|
|
}
|
|
|
|
},
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef ENABLE_EOB
|
|
|
|
// EoB options
|
|
|
|
|
|
|
|
{
|
|
|
|
GAMEOPTION_EOB_HPGRAPHS,
|
|
|
|
{
|
2012-03-25 11:47:20 +03:00
|
|
|
// I18N: HP stands for Hit Points
|
2012-02-29 17:06:13 +01:00
|
|
|
_s("HP bar graphs"),
|
|
|
|
_s("Enable hit point bar graphs"),
|
|
|
|
"hpbargraphs",
|
|
|
|
true
|
|
|
|
}
|
|
|
|
},
|
|
|
|
#endif
|
|
|
|
|
|
|
|
AD_EXTRA_GUI_OPTIONS_TERMINATOR
|
|
|
|
};
|
|
|
|
|
2010-08-24 13:11:12 +00:00
|
|
|
} // End of anonymous namespace
|
|
|
|
|
2009-01-29 22:13:01 +00:00
|
|
|
class KyraMetaEngine : public AdvancedMetaEngine {
|
2008-02-02 02:35:13 +00:00
|
|
|
public:
|
2012-02-29 17:06:13 +01:00
|
|
|
KyraMetaEngine() : AdvancedMetaEngine(adGameDescs, sizeof(KYRAGameDescription), gameList, gameGuiOptions) {
|
2011-06-11 17:52:32 +02:00
|
|
|
_md5Bytes = 1024 * 1024;
|
|
|
|
_maxScanDepth = 2;
|
|
|
|
_directoryGlobs = directoryGlobs;
|
2011-06-10 15:53:53 +02:00
|
|
|
}
|
2008-03-27 18:03:00 +00:00
|
|
|
const char *getName() const {
|
2011-05-16 23:56:12 +01:00
|
|
|
return "Kyra";
|
2008-02-02 02:35:13 +00:00
|
|
|
}
|
|
|
|
|
2009-03-05 12:04:58 +00:00
|
|
|
const char *getOriginalCopyright() const {
|
2009-05-29 12:38:41 +00:00
|
|
|
return "The Legend of Kyrandia (C) Westwood Studios"
|
|
|
|
#ifdef ENABLE_LOL
|
2011-12-26 21:19:29 +01:00
|
|
|
"\nLands of Lore (C) Westwood Studios"
|
2011-12-18 21:08:31 +01:00
|
|
|
#endif
|
|
|
|
#ifdef ENABLE_EOB
|
2011-12-26 21:19:29 +01:00
|
|
|
"\nEye of the Beholder (C) TSR, Inc., (C) Strategic Simulations, Inc."
|
2009-05-29 12:38:41 +00:00
|
|
|
#endif
|
2011-12-26 21:19:29 +01:00
|
|
|
;
|
2008-02-02 02:35:13 +00:00
|
|
|
}
|
|
|
|
|
2008-09-01 18:14:55 +00:00
|
|
|
bool hasFeature(MetaEngineFeature f) const;
|
2009-01-29 22:13:01 +00:00
|
|
|
bool createInstance(OSystem *syst, Engine **engine, const ADGameDescription *desc) const;
|
2008-09-01 18:14:55 +00:00
|
|
|
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
|
|
|
void removeSaveState(const char *target, int slot) const;
|
2008-09-16 14:10:55 +00:00
|
|
|
SaveStateDescriptor querySaveMetaInfos(const char *target, int slot) const;
|
2008-02-02 02:35:13 +00:00
|
|
|
};
|
|
|
|
|
2008-08-15 18:15:14 +00:00
|
|
|
bool KyraMetaEngine::hasFeature(MetaEngineFeature f) const {
|
|
|
|
return
|
2011-12-26 21:19:29 +01:00
|
|
|
(f == kSupportsListSaves) ||
|
|
|
|
(f == kSupportsLoadingDuringStartup) ||
|
|
|
|
(f == kSupportsDeleteSave) ||
|
|
|
|
(f == kSavesSupportMetaInfo) ||
|
|
|
|
(f == kSavesSupportThumbnail);
|
2008-11-04 16:11:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Kyra::KyraEngine_v1::hasFeature(EngineFeature f) const {
|
|
|
|
return
|
2011-12-26 21:19:29 +01:00
|
|
|
(f == kSupportsRTL) ||
|
|
|
|
(f == kSupportsLoadingDuringRuntime) ||
|
|
|
|
(f == kSupportsSavingDuringRuntime) ||
|
|
|
|
(f == kSupportsSubtitleOptions);
|
2008-08-15 18:15:14 +00:00
|
|
|
}
|
|
|
|
|
2009-01-29 22:13:01 +00:00
|
|
|
bool KyraMetaEngine::createInstance(OSystem *syst, Engine **engine, const ADGameDescription *desc) const {
|
2008-03-14 17:31:04 +00:00
|
|
|
const KYRAGameDescription *gd = (const KYRAGameDescription *)desc;
|
2007-08-13 09:00:52 +00:00
|
|
|
bool res = true;
|
2006-05-03 14:24:48 +00:00
|
|
|
|
2007-02-13 23:37:44 +00:00
|
|
|
Kyra::GameFlags flags = gd->flags;
|
2007-12-16 18:48:43 +00:00
|
|
|
|
2007-02-13 23:37:44 +00:00
|
|
|
flags.lang = gd->desc.language;
|
|
|
|
flags.platform = gd->desc.platform;
|
2007-01-25 11:32:50 +00:00
|
|
|
|
2007-01-25 21:16:57 +00:00
|
|
|
Common::Platform platform = Common::parsePlatform(ConfMan.get("platform"));
|
2007-04-15 16:41:20 +00:00
|
|
|
if (platform != Common::kPlatformUnknown)
|
2007-01-25 11:26:13 +00:00
|
|
|
flags.platform = platform;
|
|
|
|
|
2007-01-25 21:16:57 +00:00
|
|
|
if (flags.lang == Common::UNK_LANG) {
|
|
|
|
Common::Language lang = Common::parseLanguage(ConfMan.get("language"));
|
2007-04-15 16:41:20 +00:00
|
|
|
if (lang != Common::UNK_LANG)
|
2007-01-25 11:26:13 +00:00
|
|
|
flags.lang = lang;
|
2007-04-15 16:41:20 +00:00
|
|
|
else
|
2007-01-25 21:16:57 +00:00
|
|
|
flags.lang = Common::EN_ANY;
|
2007-01-25 11:26:13 +00:00
|
|
|
}
|
2007-01-25 00:07:48 +00:00
|
|
|
|
2007-08-13 09:00:52 +00:00
|
|
|
switch (flags.gameID) {
|
|
|
|
case Kyra::GI_KYRA1:
|
2008-05-11 22:32:37 +00:00
|
|
|
*engine = new Kyra::KyraEngine_LoK(syst, flags);
|
2007-08-13 09:00:52 +00:00
|
|
|
break;
|
|
|
|
case Kyra::GI_KYRA2:
|
2008-05-02 14:46:30 +00:00
|
|
|
*engine = new Kyra::KyraEngine_HoF(syst, flags);
|
2007-08-13 09:00:52 +00:00
|
|
|
break;
|
|
|
|
case Kyra::GI_KYRA3:
|
2008-05-02 14:46:30 +00:00
|
|
|
*engine = new Kyra::KyraEngine_MR(syst, flags);
|
2007-08-13 09:00:52 +00:00
|
|
|
break;
|
2009-03-04 00:45:34 +00:00
|
|
|
#ifdef ENABLE_LOL
|
2008-07-31 10:47:15 +00:00
|
|
|
case Kyra::GI_LOL:
|
|
|
|
*engine = new Kyra::LoLEngine(syst, flags);
|
|
|
|
break;
|
2009-03-04 00:45:34 +00:00
|
|
|
#endif // ENABLE_LOL
|
2011-12-11 01:57:03 +01:00
|
|
|
#ifdef ENABLE_EOB
|
|
|
|
case Kyra::GI_EOB1:
|
2011-12-12 16:23:01 +01:00
|
|
|
*engine = new Kyra::EoBEngine(syst, flags);
|
2011-12-11 01:57:03 +01:00
|
|
|
break;
|
|
|
|
case Kyra::GI_EOB2:
|
2012-02-23 21:36:14 +01:00
|
|
|
if (Common::parseRenderMode(ConfMan.get("render_mode")) == Common::kRenderEGA)
|
|
|
|
flags.useHiRes = true;
|
2011-12-11 01:57:03 +01:00
|
|
|
*engine = new Kyra::DarkMoonEngine(syst, flags);
|
|
|
|
break;
|
|
|
|
#endif // ENABLE_EOB
|
2007-08-13 09:00:52 +00:00
|
|
|
default:
|
|
|
|
res = false;
|
2008-03-27 18:03:00 +00:00
|
|
|
warning("Kyra engine: unknown gameID");
|
2007-08-13 09:00:52 +00:00
|
|
|
}
|
2006-09-18 17:30:51 +00:00
|
|
|
|
2007-08-13 09:00:52 +00:00
|
|
|
return res;
|
2006-05-03 14:24:48 +00:00
|
|
|
}
|
|
|
|
|
2008-03-27 18:03:00 +00:00
|
|
|
SaveStateList KyraMetaEngine::listSaves(const char *target) const {
|
|
|
|
Common::SaveFileManager *saveFileMan = g_system->getSavefileManager();
|
2008-05-11 23:16:50 +00:00
|
|
|
Kyra::KyraEngine_v1::SaveHeader header;
|
2008-03-27 18:03:00 +00:00
|
|
|
Common::String pattern = target;
|
|
|
|
pattern += ".???";
|
|
|
|
|
2010-03-18 15:54:40 +00:00
|
|
|
Common::StringArray filenames;
|
2009-05-29 14:38:22 +00:00
|
|
|
filenames = saveFileMan->listSavefiles(pattern);
|
2011-12-26 21:19:29 +01:00
|
|
|
Common::sort(filenames.begin(), filenames.end()); // Sort (hopefully ensuring we are sorted numerically..)
|
2008-03-27 18:03:00 +00:00
|
|
|
|
|
|
|
SaveStateList saveList;
|
2010-03-18 15:54:40 +00:00
|
|
|
for (Common::StringArray::const_iterator file = filenames.begin(); file != filenames.end(); ++file) {
|
2011-08-21 20:26:07 +02:00
|
|
|
// Skip automatic final saves made by EOB for the purpose of party transfer
|
|
|
|
if (!scumm_stricmp(file->c_str() + file->size() - 3, "fin"))
|
|
|
|
continue;
|
2011-12-11 12:29:09 +01:00
|
|
|
|
2008-03-27 18:03:00 +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-09-16 14:10:55 +00:00
|
|
|
if (slotNum >= 0 && slotNum <= 999) {
|
2009-05-29 14:38:22 +00:00
|
|
|
Common::InSaveFile *in = saveFileMan->openForLoading(*file);
|
2008-03-27 18:03:00 +00:00
|
|
|
if (in) {
|
2008-09-12 12:25:56 +00:00
|
|
|
if (Kyra::KyraEngine_v1::readSaveHeader(in, false, header) == Kyra::KyraEngine_v1::kRSHENoError) {
|
2009-05-29 14:58:08 +00:00
|
|
|
// WORKAROUND: Old savegames are using 'German' as description for kyra3 restart game save (slot 0),
|
|
|
|
// since that looks odd we replace it by "New Game".
|
2008-09-12 12:25:56 +00:00
|
|
|
if (slotNum == 0 && header.gameID == Kyra::GI_KYRA3)
|
|
|
|
header.description = "New Game";
|
|
|
|
|
2008-11-05 15:41:12 +00:00
|
|
|
saveList.push_back(SaveStateDescriptor(slotNum, header.description));
|
2008-09-12 12:25:56 +00:00
|
|
|
}
|
2008-03-27 18:03:00 +00:00
|
|
|
delete in;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return saveList;
|
|
|
|
}
|
|
|
|
|
2011-12-26 21:19:29 +01:00
|
|
|
int KyraMetaEngine::getMaximumSaveSlot() const {
|
|
|
|
return 999;
|
|
|
|
}
|
2008-11-09 16:13:34 +00:00
|
|
|
|
2008-09-11 19:47:45 +00:00
|
|
|
void KyraMetaEngine::removeSaveState(const char *target, int slot) const {
|
2009-05-29 14:58:08 +00:00
|
|
|
// In Kyra games slot 0 can't be deleted, it's for restarting the game(s).
|
|
|
|
// An exception makes Lands of Lore here, it does not have any way to restart the
|
|
|
|
// game except via its main menu.
|
2011-08-20 13:24:36 +02:00
|
|
|
if (slot == 0 && !ConfMan.getDomain(target)->getVal("gameid").equalsIgnoreCase("lol") && !ConfMan.getDomain(target)->getVal("gameid").equalsIgnoreCase("eob") && !ConfMan.getDomain(target)->getVal("gameid").equalsIgnoreCase("eob2"))
|
2008-09-14 14:01:24 +00:00
|
|
|
return;
|
|
|
|
|
2008-09-14 21:41:27 +00:00
|
|
|
Common::String filename = Kyra::KyraEngine_v1::getSavegameFilename(target, slot);
|
2009-05-29 14:38:22 +00:00
|
|
|
g_system->getSavefileManager()->removeSavefile(filename);
|
2008-09-11 19:47:45 +00:00
|
|
|
}
|
|
|
|
|
2008-09-16 14:10:55 +00:00
|
|
|
SaveStateDescriptor KyraMetaEngine::querySaveMetaInfos(const char *target, int slot) const {
|
2008-09-14 21:41:27 +00:00
|
|
|
Common::String filename = Kyra::KyraEngine_v1::getSavegameFilename(target, slot);
|
2009-05-29 14:38:22 +00:00
|
|
|
Common::InSaveFile *in = g_system->getSavefileManager()->openForLoading(filename);
|
2011-12-12 22:23:30 +01:00
|
|
|
const bool nonKyraGame = ConfMan.getDomain(target)->getVal("gameid").equalsIgnoreCase("lol") || ConfMan.getDomain(target)->getVal("gameid").equalsIgnoreCase("eob") || ConfMan.getDomain(target)->getVal("gameid").equalsIgnoreCase("eob2");
|
2008-09-14 21:00:24 +00:00
|
|
|
|
2008-09-16 14:10:55 +00:00
|
|
|
if (in) {
|
|
|
|
Kyra::KyraEngine_v1::SaveHeader header;
|
|
|
|
Kyra::KyraEngine_v1::kReadSaveHeaderError error;
|
|
|
|
|
|
|
|
error = Kyra::KyraEngine_v1::readSaveHeader(in, true, header);
|
|
|
|
delete in;
|
2009-01-01 15:06:43 +00:00
|
|
|
|
2008-09-16 14:10:55 +00:00
|
|
|
if (error == Kyra::KyraEngine_v1::kRSHENoError) {
|
2008-11-05 15:41:12 +00:00
|
|
|
SaveStateDescriptor desc(slot, header.description);
|
2008-09-16 14:10:55 +00:00
|
|
|
|
2009-05-29 14:58:08 +00:00
|
|
|
// Slot 0 is used for the 'restart game' save in all three Kyrandia games, thus
|
|
|
|
// we prevent it from being deleted.
|
2011-08-20 13:24:36 +02:00
|
|
|
desc.setDeletableFlag(slot != 0 || nonKyraGame);
|
2009-05-29 14:58:08 +00:00
|
|
|
|
|
|
|
// We don't allow quick saves (slot 990 till 998) to be overwritten.
|
2011-10-01 17:03:29 +02:00
|
|
|
// The same goes for the 'Autosave', which is slot 999. Slot 0 will also
|
|
|
|
// be protected in Kyra 1-3, since it's the 'restart game' save.
|
2011-08-20 13:24:36 +02:00
|
|
|
desc.setWriteProtectedFlag((slot == 0 && !nonKyraGame) || slot >= 990);
|
2008-09-16 14:10:55 +00:00
|
|
|
desc.setThumbnail(header.thumbnail);
|
|
|
|
|
|
|
|
return desc;
|
|
|
|
}
|
|
|
|
}
|
2009-01-01 15:06:43 +00:00
|
|
|
|
2011-10-01 17:03:29 +02:00
|
|
|
SaveStateDescriptor desc(slot, Common::String());
|
|
|
|
|
|
|
|
// We don't allow quick saves (slot 990 till 998) to be overwritten.
|
|
|
|
// The same goes for the 'Autosave', which is slot 999. Slot 0 will also
|
|
|
|
// be protected in Kyra 1-3, since it's the 'restart game' save.
|
2011-08-20 13:24:36 +02:00
|
|
|
desc.setWriteProtectedFlag((slot == 0 && !nonKyraGame) || slot >= 990);
|
2011-10-01 17:03:29 +02:00
|
|
|
|
|
|
|
return desc;
|
2008-09-14 21:00:24 +00:00
|
|
|
}
|
|
|
|
|
2008-05-06 03:00:26 +00:00
|
|
|
#if PLUGIN_ENABLED_DYNAMIC(KYRA)
|
|
|
|
REGISTER_PLUGIN_DYNAMIC(KYRA, PLUGIN_TYPE_ENGINE, KyraMetaEngine);
|
|
|
|
#else
|
|
|
|
REGISTER_PLUGIN_STATIC(KYRA, PLUGIN_TYPE_ENGINE, KyraMetaEngine);
|
|
|
|
#endif
|