2009-05-26 14:13:08 +00:00
|
|
|
/* Residual - A 3D game interpreter
|
2009-05-24 21:31:05 +00:00
|
|
|
*
|
|
|
|
* Residual is the legal property of its developers, whose names
|
|
|
|
* are too numerous to list here. Please refer to the AUTHORS
|
|
|
|
* 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.
|
2009-10-04 10:58:28 +00:00
|
|
|
*
|
2009-05-24 21:31:05 +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.
|
2009-10-04 10:58:28 +00:00
|
|
|
*
|
2009-05-24 21:31:05 +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.
|
|
|
|
*
|
|
|
|
* $URL$
|
|
|
|
* $Id$
|
|
|
|
*/
|
|
|
|
|
|
|
|
#if defined(WIN32)
|
|
|
|
#include <windows.h>
|
|
|
|
#include <direct.h>
|
|
|
|
// winnt.h defines ARRAYSIZE, but we want our own one...
|
|
|
|
#undef ARRAYSIZE
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "engines/engine.h"
|
|
|
|
#include "common/config-manager.h"
|
|
|
|
#include "common/events.h"
|
|
|
|
#include "common/file.h"
|
|
|
|
#include "common/timer.h"
|
|
|
|
#include "common/savefile.h"
|
|
|
|
#include "common/system.h"
|
2009-10-04 10:58:28 +00:00
|
|
|
#include "gui/message.h"
|
|
|
|
#include "gui/GuiManager.h"
|
2009-05-25 19:21:58 +00:00
|
|
|
#include "sound/mixer.h"
|
2009-10-04 10:58:28 +00:00
|
|
|
#include "engines/dialogs.h"
|
2009-05-24 21:31:05 +00:00
|
|
|
#include "engines/metaengine.h"
|
|
|
|
|
|
|
|
#ifdef _WIN32_WCE
|
|
|
|
extern bool isSmartphone(void);
|
|
|
|
#endif
|
|
|
|
|
2009-10-04 10:58:28 +00:00
|
|
|
// FIXME: HACK for MidiEmu & error()
|
|
|
|
Engine *g_engine = 0;
|
|
|
|
|
|
|
|
|
2009-05-24 21:31:05 +00:00
|
|
|
Engine::Engine(OSystem *syst)
|
|
|
|
: _system(syst),
|
|
|
|
_mixer(_system->getMixer()),
|
|
|
|
_timer(_system->getTimerManager()),
|
|
|
|
_eventMan(_system->getEventManager()),
|
|
|
|
_saveFileMan(_system->getSavefileManager()),
|
|
|
|
_targetName(ConfMan.getActiveDomainName()),
|
|
|
|
_gameDataDir(ConfMan.get("path")),
|
2009-10-04 10:58:28 +00:00
|
|
|
_pauseLevel(0),
|
|
|
|
_mainMenuDialog(NULL) {
|
|
|
|
|
|
|
|
g_engine = this;
|
2009-05-24 21:31:05 +00:00
|
|
|
|
|
|
|
// FIXME: Get rid of the following again. It is only here temporarily.
|
|
|
|
// We really should never run with a non-working Mixer, so ought to handle
|
|
|
|
// this at a much earlier stage. If we *really* want to support systems
|
|
|
|
// without a working mixer, then we need more work. E.g. we could modify the
|
|
|
|
// Mixer to immediately drop any streams passed to it. This way, at least
|
|
|
|
// we don't crash because heaps of (sound) memory get allocated but never
|
|
|
|
// freed. Of course, there still would be problems with many games...
|
|
|
|
if (!_mixer->isReady())
|
|
|
|
warning("Sound initialization failed. This may cause severe problems in some games.");
|
|
|
|
}
|
|
|
|
|
|
|
|
Engine::~Engine() {
|
|
|
|
_mixer->stopAll();
|
2009-10-04 10:58:28 +00:00
|
|
|
|
|
|
|
delete _mainMenuDialog;
|
|
|
|
g_engine = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void GUIErrorMessage(const Common::String msg) {
|
|
|
|
g_system->setWindowCaption("Error");
|
|
|
|
g_system->launcherInitSize(320, 200);
|
|
|
|
GUI::MessageDialog dialog(msg);
|
|
|
|
dialog.runModal();
|
|
|
|
error("%s", msg.c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
void Engine::checkCD() {
|
|
|
|
#if defined (WIN32) && !defined(_WIN32_WCE) && !defined(__SYMBIAN32__)
|
|
|
|
// It is a known bug under Windows that games that play CD audio cause
|
|
|
|
// ScummVM to crash if the data files are read from the same CD. Check
|
|
|
|
// if this appears to be the case and issue a warning.
|
|
|
|
|
|
|
|
// If we can find a compressed audio track, then it should be ok even
|
|
|
|
// if it's running from CD.
|
|
|
|
|
|
|
|
#ifdef USE_VORBIS
|
|
|
|
if (Common::File::exists("track1.ogg") ||
|
|
|
|
Common::File::exists("track01.ogg"))
|
|
|
|
return;
|
|
|
|
#endif
|
|
|
|
#ifdef USE_FLAC
|
|
|
|
if (Common::File::exists("track1.fla") ||
|
|
|
|
Common::File::exists("track1.flac") ||
|
|
|
|
Common::File::exists("track01.fla") ||
|
|
|
|
Common::File::exists("track01.flac"))
|
|
|
|
return;
|
|
|
|
#endif
|
|
|
|
#ifdef USE_MAD
|
|
|
|
if (Common::File::exists("track1.mp3") ||
|
|
|
|
Common::File::exists("track01.mp3"))
|
|
|
|
return;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
char buffer[MAXPATHLEN];
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (_gameDataDir.getPath().empty()) {
|
|
|
|
// That's it! I give up!
|
|
|
|
if (getcwd(buffer, MAXPATHLEN) == NULL)
|
|
|
|
return;
|
|
|
|
} else
|
|
|
|
strncpy(buffer, _gameDataDir.getPath().c_str(), MAXPATHLEN);
|
|
|
|
|
|
|
|
for (i = 0; i < MAXPATHLEN - 1; i++) {
|
|
|
|
if (buffer[i] == '\\')
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
buffer[i + 1] = 0;
|
|
|
|
|
|
|
|
if (GetDriveType(buffer) == DRIVE_CDROM) {
|
|
|
|
GUI::MessageDialog dialog(
|
|
|
|
"You appear to be playing this game directly\n"
|
|
|
|
"from the CD. This is known to cause problems,\n"
|
|
|
|
"and it is therefore recommended that you copy\n"
|
|
|
|
"the data files to your hard disk instead.\n"
|
|
|
|
"See the README file for details.", "OK");
|
|
|
|
dialog.runModal();
|
|
|
|
} else {
|
|
|
|
// If we reached here, the game has audio tracks,
|
|
|
|
// it's not ran from the CD and the tracks have not
|
|
|
|
// been ripped.
|
|
|
|
GUI::MessageDialog dialog(
|
|
|
|
"This game has audio tracks in its disk. These\n"
|
|
|
|
"tracks need to be ripped from the disk using\n"
|
|
|
|
"an appropriate CD audio extracting tool in\n"
|
|
|
|
"order to listen to the game's music.\n"
|
|
|
|
"See the README file for details.", "OK");
|
|
|
|
dialog.runModal();
|
|
|
|
}
|
|
|
|
#endif
|
2009-05-24 21:31:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Engine::shouldPerformAutoSave(int lastSaveTime) {
|
|
|
|
const int diff = _system->getMillis() - lastSaveTime;
|
|
|
|
const int autosavePeriod = ConfMan.getInt("autosave_period");
|
|
|
|
return autosavePeriod != 0 && diff > autosavePeriod * 1000;
|
|
|
|
}
|
|
|
|
|
2009-10-04 10:58:28 +00:00
|
|
|
void Engine::errorString(const char *buf1, char *buf2, int size) {
|
|
|
|
strncpy(buf2, buf1, size);
|
|
|
|
if (size > 0)
|
|
|
|
buf2[size-1] = '\0';
|
|
|
|
}
|
|
|
|
|
2009-05-24 21:31:05 +00:00
|
|
|
void Engine::pauseEngine(bool pause) {
|
|
|
|
assert((pause && _pauseLevel >= 0) || (!pause && _pauseLevel));
|
|
|
|
|
|
|
|
if (pause)
|
|
|
|
_pauseLevel++;
|
|
|
|
else
|
|
|
|
_pauseLevel--;
|
|
|
|
|
|
|
|
if (_pauseLevel == 1) {
|
|
|
|
pauseEngineIntern(true);
|
|
|
|
} else if (_pauseLevel == 0) {
|
|
|
|
pauseEngineIntern(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Engine::pauseEngineIntern(bool pause) {
|
|
|
|
// By default, just (un)pause all digital sounds
|
|
|
|
_mixer->pauseAll(pause);
|
|
|
|
}
|
|
|
|
|
2009-10-04 10:58:28 +00:00
|
|
|
void Engine::openMainMenuDialog() {
|
|
|
|
if (!_mainMenuDialog)
|
|
|
|
_mainMenuDialog = new MainMenuDialog(this);
|
|
|
|
runDialog(*_mainMenuDialog);
|
|
|
|
syncSoundSettings();
|
|
|
|
}
|
|
|
|
|
|
|
|
int Engine::runDialog(GUI::Dialog &dialog) {
|
|
|
|
pauseEngine(true);
|
|
|
|
int result = dialog.runModal();
|
|
|
|
pauseEngine(false);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2009-05-24 21:31:05 +00:00
|
|
|
void Engine::syncSoundSettings() {
|
|
|
|
|
|
|
|
// Sync the engine with the config manager
|
|
|
|
int soundVolumeMusic = ConfMan.getInt("music_volume");
|
|
|
|
int soundVolumeSFX = ConfMan.getInt("sfx_volume");
|
|
|
|
int soundVolumeSpeech = ConfMan.getInt("speech_volume");
|
|
|
|
|
2009-06-18 11:52:26 +00:00
|
|
|
bool mute = false;
|
|
|
|
if (ConfMan.hasKey("mute"))
|
|
|
|
mute = ConfMan.getBool("mute");
|
|
|
|
|
|
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kMusicSoundType, (mute ? 0 : soundVolumeMusic));
|
|
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kSFXSoundType, (mute ? 0 : soundVolumeSFX));
|
|
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kSpeechSoundType, (mute ? 0 : soundVolumeSpeech));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Engine::flipMute() {
|
2009-10-05 08:33:46 +00:00
|
|
|
// Mute will be set to true by default here. This has two reasons:
|
|
|
|
// - if the game already has an "mute" config entry, it will be overwritten anyway.
|
|
|
|
// - if it does not have a "mute" config entry, the sound is unmuted currently and should be muted now.
|
|
|
|
bool mute = true;
|
2009-06-18 11:52:26 +00:00
|
|
|
|
|
|
|
if (ConfMan.hasKey("mute")) {
|
|
|
|
mute = !ConfMan.getBool("mute");
|
|
|
|
}
|
|
|
|
|
|
|
|
ConfMan.setBool("mute", mute);
|
|
|
|
|
|
|
|
syncSoundSettings();
|
2009-05-24 21:31:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Common::Error Engine::loadGameState(int slot) {
|
|
|
|
// Do nothing by default
|
|
|
|
return Common::kNoError;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Engine::canLoadGameStateCurrently() {
|
|
|
|
// Do not allow loading by default
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Common::Error Engine::saveGameState(int slot, const char *desc) {
|
|
|
|
// Do nothing by default
|
|
|
|
return Common::kNoError;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Engine::canSaveGameStateCurrently() {
|
|
|
|
// Do not allow saving by default
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Engine::quitGame() {
|
|
|
|
Common::Event event;
|
|
|
|
|
|
|
|
event.type = Common::EVENT_QUIT;
|
|
|
|
g_system->getEventManager()->pushEvent(event);
|
|
|
|
}
|
|
|
|
|
2009-06-18 11:52:26 +00:00
|
|
|
bool Engine::shouldQuit() {
|
|
|
|
Common::EventManager *eventMan = g_system->getEventManager();
|
2009-10-04 10:58:28 +00:00
|
|
|
return (eventMan->shouldQuit() || eventMan->shouldRTL());
|
2009-06-18 11:52:26 +00:00
|
|
|
}
|
|
|
|
|
2009-05-24 21:31:05 +00:00
|
|
|
/*
|
|
|
|
EnginePlugin *Engine::getMetaEnginePlugin() const {
|
|
|
|
|
|
|
|
const EnginePlugin *plugin = 0;
|
|
|
|
Common::String gameid = ConfMan.get("gameid");
|
|
|
|
gameid.toLowercase();
|
|
|
|
EngineMan.findGame(gameid, &plugin);
|
|
|
|
return plugin;
|
|
|
|
}
|
|
|
|
|
|
|
|
*/
|