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.
|
2002-08-18 18:04:07 +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
|
2005-10-18 01:30:26 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2014-02-18 02:34:20 +01:00
|
|
|
*
|
2002-08-18 18:04:07 +00:00
|
|
|
*/
|
|
|
|
|
2011-05-03 14:30:25 +02:00
|
|
|
#define FORBIDDEN_SYMBOL_EXCEPTION_getcwd
|
|
|
|
|
2019-11-10 16:34:25 +00:00
|
|
|
#if defined(WIN32) && !defined(__SYMBIAN32__)
|
2010-06-26 21:55:52 +00:00
|
|
|
#define WIN32_LEAN_AND_MEAN
|
2007-03-17 15:45:05 +00:00
|
|
|
#include <windows.h>
|
|
|
|
#include <direct.h>
|
|
|
|
#endif
|
|
|
|
|
2006-09-23 00:42:35 +00:00
|
|
|
#include "engines/engine.h"
|
2010-04-11 18:30:42 +00:00
|
|
|
#include "engines/dialogs.h"
|
2017-10-01 16:23:22 -05:00
|
|
|
#include "engines/util.h"
|
2020-02-02 11:11:41 -08:00
|
|
|
#include "engines/metaengine.h"
|
2010-04-11 18:30:42 +00:00
|
|
|
|
2003-10-08 21:59:23 +00:00
|
|
|
#include "common/config-manager.h"
|
2008-10-06 12:48:52 +00:00
|
|
|
#include "common/events.h"
|
2011-04-28 22:32:12 +10:00
|
|
|
#include "common/file.h"
|
2004-12-27 22:08:20 +00:00
|
|
|
#include "common/system.h"
|
2010-05-05 17:54:12 +00:00
|
|
|
#include "common/str.h"
|
2020-06-30 03:40:13 +05:30
|
|
|
#include "common/ustr.h"
|
2011-04-24 11:34:27 +03:00
|
|
|
#include "common/error.h"
|
|
|
|
#include "common/list.h"
|
2015-11-07 22:35:10 +01:00
|
|
|
#include "common/memstream.h"
|
2020-02-02 11:11:41 -08:00
|
|
|
#include "common/savefile.h"
|
2011-04-24 11:34:27 +03:00
|
|
|
#include "common/scummsys.h"
|
2011-08-05 21:16:38 +01:00
|
|
|
#include "common/taskbar.h"
|
2011-04-24 11:34:27 +03:00
|
|
|
#include "common/textconsole.h"
|
2011-06-13 22:19:18 +01:00
|
|
|
#include "common/translation.h"
|
2014-12-30 10:47:51 +01:00
|
|
|
#include "common/singleton.h"
|
2010-04-11 18:30:42 +00:00
|
|
|
|
2017-08-14 17:11:33 +02:00
|
|
|
#include "backends/keymapper/action.h"
|
2011-12-30 11:50:15 -06:00
|
|
|
#include "backends/keymapper/keymapper.h"
|
2015-12-24 00:07:22 +01:00
|
|
|
#include "base/version.h"
|
2011-12-30 11:50:15 -06:00
|
|
|
|
2015-11-11 21:36:27 +01:00
|
|
|
#include "gui/gui-manager.h"
|
2009-11-24 22:11:07 +00:00
|
|
|
#include "gui/debugger.h"
|
2011-04-24 11:34:27 +03:00
|
|
|
#include "gui/dialog.h"
|
2005-06-14 08:54:11 +00:00
|
|
|
#include "gui/message.h"
|
2020-02-02 16:19:29 -08:00
|
|
|
#include "gui/saveload.h"
|
2010-04-11 18:30:42 +00:00
|
|
|
|
2011-02-09 01:09:01 +00:00
|
|
|
#include "audio/mixer.h"
|
2010-04-11 18:30:42 +00:00
|
|
|
|
|
|
|
#include "graphics/cursorman.h"
|
2015-12-24 00:07:22 +01:00
|
|
|
#include "graphics/fontman.h"
|
2011-04-24 11:34:27 +03:00
|
|
|
#include "graphics/pixelformat.h"
|
2015-11-07 22:35:10 +01:00
|
|
|
#include "image/bmp.h"
|
2002-08-18 18:04:07 +00:00
|
|
|
|
2019-07-22 19:42:30 +02:00
|
|
|
#ifdef USE_TTS
|
|
|
|
#include "common/text-to-speech.h"
|
|
|
|
#endif
|
|
|
|
|
2010-01-25 03:04:58 +00:00
|
|
|
// FIXME: HACK for error()
|
2003-03-07 15:38:11 +00:00
|
|
|
Engine *g_engine = 0;
|
2002-08-18 18:04:07 +00:00
|
|
|
|
2009-11-24 22:11:07 +00:00
|
|
|
// Output formatter for debug() and error() which invokes
|
|
|
|
// the errorString method of the active engine, if any.
|
|
|
|
static void defaultOutputFormatter(char *dst, const char *src, size_t dstSize) {
|
|
|
|
if (g_engine) {
|
|
|
|
g_engine->errorString(src, dst, dstSize);
|
|
|
|
} else {
|
2010-05-05 17:54:12 +00:00
|
|
|
Common::strlcpy(dst, src, dstSize);
|
2009-11-24 22:11:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void defaultErrorHandler(const char *msg) {
|
|
|
|
// Unless this error -originated- within the debugger itself, we
|
|
|
|
// now invoke the debugger, if available / supported.
|
|
|
|
if (g_engine) {
|
2020-02-09 11:06:20 -08:00
|
|
|
GUI::Debugger *debugger = g_engine->getOrCreateDebugger();
|
2011-08-06 15:27:43 -04:00
|
|
|
|
|
|
|
#if defined(USE_TASKBAR)
|
|
|
|
g_system->getTaskbarManager()->notifyError();
|
|
|
|
#endif
|
|
|
|
|
2010-07-17 18:38:42 +00:00
|
|
|
if (debugger && !debugger->isActive()) {
|
2009-11-24 22:11:07 +00:00
|
|
|
debugger->attach(msg);
|
|
|
|
debugger->onFrame();
|
|
|
|
}
|
2011-08-05 21:16:38 +01:00
|
|
|
|
2011-08-06 15:27:43 -04:00
|
|
|
|
2011-08-05 21:16:38 +01:00
|
|
|
#if defined(USE_TASKBAR)
|
2011-08-06 15:27:43 -04:00
|
|
|
g_system->getTaskbarManager()->clearError();
|
2011-08-05 21:16:38 +01:00
|
|
|
#endif
|
|
|
|
|
2009-11-24 22:11:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-30 10:47:51 +01:00
|
|
|
// Chained games manager
|
|
|
|
|
|
|
|
ChainedGamesManager::ChainedGamesManager() {
|
|
|
|
clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ChainedGamesManager::clear() {
|
|
|
|
_chainedGames.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ChainedGamesManager::push(const Common::String target, const int slot) {
|
|
|
|
Game game;
|
|
|
|
game.target = target;
|
|
|
|
game.slot = slot;
|
|
|
|
_chainedGames.push(game);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ChainedGamesManager::pop(Common::String &target, int &slot) {
|
|
|
|
if (_chainedGames.empty()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
Game game = _chainedGames.pop();
|
|
|
|
target = game.target;
|
|
|
|
slot = game.slot;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-07-10 21:17:41 +02:00
|
|
|
namespace Common {
|
|
|
|
DECLARE_SINGLETON(ChainedGamesManager);
|
|
|
|
}
|
|
|
|
|
2003-11-10 01:04:12 +00:00
|
|
|
Engine::Engine(OSystem *syst)
|
2006-04-15 20:36:41 +00:00
|
|
|
: _system(syst),
|
2006-10-21 12:03:43 +00:00
|
|
|
_mixer(_system->getMixer()),
|
|
|
|
_timer(_system->getTimerManager()),
|
2007-04-01 17:36:13 +00:00
|
|
|
_eventMan(_system->getEventManager()),
|
2006-10-21 12:03:43 +00:00
|
|
|
_saveFileMan(_system->getSavefileManager()),
|
2007-02-12 19:47:42 +00:00
|
|
|
_targetName(ConfMan.getActiveDomainName()),
|
2008-06-24 21:15:30 +00:00
|
|
|
_pauseLevel(0),
|
2010-10-27 22:37:51 +00:00
|
|
|
_pauseStartTime(0),
|
2011-06-25 15:41:58 +02:00
|
|
|
_saveSlotToLoad(-1),
|
2010-10-27 22:37:51 +00:00
|
|
|
_engineStartTime(_system->getMillis()),
|
2020-02-08 20:33:53 -08:00
|
|
|
_mainMenuDialog(NULL),
|
2020-02-04 22:25:13 -08:00
|
|
|
_debugger(NULL),
|
|
|
|
_autosaveInterval(ConfMan.getInt("autosave_period")),
|
|
|
|
_lastAutosaveTime(_system->getMillis()) {
|
2005-11-23 19:11:33 +00:00
|
|
|
|
2006-10-21 12:03:43 +00:00
|
|
|
g_engine = this;
|
2009-11-24 22:11:07 +00:00
|
|
|
Common::setErrorOutputFormatter(defaultOutputFormatter);
|
|
|
|
Common::setErrorHandler(defaultErrorHandler);
|
2008-02-15 17:01:35 +00:00
|
|
|
|
2010-04-11 19:04:24 +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...
|
2008-02-15 17:01:35 +00:00
|
|
|
if (!_mixer->isReady())
|
2010-09-18 10:55:16 +00:00
|
|
|
warning("Sound initialization failed. This may cause severe problems in some games");
|
2010-04-11 18:30:42 +00:00
|
|
|
|
2010-04-11 19:04:24 +00:00
|
|
|
// Setup a dummy cursor and palette, so that all engines can use
|
|
|
|
// CursorMan.replace without having any headaches about memory leaks.
|
|
|
|
//
|
|
|
|
// If an engine only used CursorMan.replaceCursor and no cursor has
|
|
|
|
// been setup before, then replaceCursor just uses pushCursor. This
|
|
|
|
// means that that the engine's cursor is never again removed from
|
|
|
|
// CursorMan. Hence we setup a fake cursor here and remove it again
|
|
|
|
// in the destructor.
|
2010-04-11 18:30:42 +00:00
|
|
|
CursorMan.pushCursor(NULL, 0, 0, 0, 0, 0);
|
2010-04-11 19:04:24 +00:00
|
|
|
// Note: Using this dummy palette will actually disable cursor
|
|
|
|
// palettes till the user enables it again.
|
2010-04-11 18:30:42 +00:00
|
|
|
CursorMan.pushCursorPalette(NULL, 0, 0);
|
2002-08-18 18:04:07 +00:00
|
|
|
}
|
|
|
|
|
2003-03-06 16:27:06 +00:00
|
|
|
Engine::~Engine() {
|
2007-02-20 18:50:17 +00:00
|
|
|
_mixer->stopAll();
|
2009-01-01 15:06:43 +00:00
|
|
|
|
2020-02-08 20:33:53 -08:00
|
|
|
delete _debugger;
|
2008-06-26 23:22:28 +00:00
|
|
|
delete _mainMenuDialog;
|
2004-11-19 19:50:22 +00:00
|
|
|
g_engine = NULL;
|
2010-04-11 18:30:42 +00:00
|
|
|
|
|
|
|
// Remove our cursors again to prevent memory leaks
|
|
|
|
CursorMan.popCursor();
|
|
|
|
CursorMan.popCursorPalette();
|
2002-08-18 18:04:07 +00:00
|
|
|
}
|
2002-08-18 18:39:42 +00:00
|
|
|
|
2014-01-22 18:07:06 +01:00
|
|
|
void Engine::initializePath(const Common::FSNode &gamePath) {
|
|
|
|
SearchMan.addDirectory(gamePath.getPath(), gamePath, 0, 4);
|
|
|
|
}
|
|
|
|
|
2017-10-01 00:56:01 -05:00
|
|
|
void initCommonGFX() {
|
2006-04-15 20:36:41 +00:00
|
|
|
const Common::ConfigManager::Domain *gameDomain = ConfMan.getActiveDomain();
|
2006-04-15 13:12:03 +00:00
|
|
|
|
2017-10-01 16:23:22 -05:00
|
|
|
// Any global or command line settings already have been applied at the time
|
|
|
|
// we get here, so we only do something if the game domain overrides those
|
|
|
|
// values
|
|
|
|
if (gameDomain) {
|
|
|
|
if (gameDomain->contains("aspect_ratio"))
|
|
|
|
g_system->setFeatureState(OSystem::kFeatureAspectRatioCorrection, ConfMan.getBool("aspect_ratio"));
|
2006-04-15 13:12:03 +00:00
|
|
|
|
2017-10-01 16:23:22 -05:00
|
|
|
if (gameDomain->contains("fullscreen"))
|
|
|
|
g_system->setFeatureState(OSystem::kFeatureFullscreenMode, ConfMan.getBool("fullscreen"));
|
2005-07-30 21:11:48 +00:00
|
|
|
|
2017-10-01 16:23:22 -05:00
|
|
|
if (gameDomain->contains("filtering"))
|
|
|
|
g_system->setFeatureState(OSystem::kFeatureFilteringMode, ConfMan.getBool("filtering"));
|
2018-10-22 18:18:27 +02:00
|
|
|
|
|
|
|
if (gameDomain->contains("stretch_mode"))
|
|
|
|
g_system->setStretchMode(ConfMan.get("stretch_mode").c_str());
|
2020-01-29 23:13:32 +00:00
|
|
|
|
|
|
|
if (gameDomain->contains("shader"))
|
2020-01-31 23:11:09 +00:00
|
|
|
g_system->setShader(ConfMan.get("shader").c_str());
|
2017-10-01 16:23:22 -05:00
|
|
|
}
|
2008-10-02 17:48:01 +00:00
|
|
|
}
|
2009-06-30 08:25:08 +00:00
|
|
|
|
2015-11-09 00:44:57 +01:00
|
|
|
// Please leave the splash screen in working order for your releases, even if they're commercial.
|
2015-11-07 22:35:10 +01:00
|
|
|
// This is a proper and good way to show your appreciation for our hard work over these years.
|
|
|
|
bool splash = false;
|
|
|
|
|
|
|
|
#include "logo_data.h"
|
|
|
|
|
|
|
|
void splashScreen() {
|
|
|
|
Common::MemoryReadStream stream(logo_data, ARRAYSIZE(logo_data));
|
|
|
|
|
|
|
|
Image::BitmapDecoder bitmap;
|
|
|
|
|
|
|
|
if (!bitmap.loadStream(stream)) {
|
|
|
|
warning("Error loading logo file");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_system->showOverlay();
|
|
|
|
|
2015-12-22 19:09:13 +01:00
|
|
|
// Fill with orange
|
2015-11-07 22:35:10 +01:00
|
|
|
Graphics::Surface screen;
|
|
|
|
screen.create(g_system->getOverlayWidth(), g_system->getOverlayHeight(), g_system->getOverlayFormat());
|
2019-03-30 23:21:55 +00:00
|
|
|
screen.fillRect(Common::Rect(screen.w, screen.h), screen.format.ARGBToColor(0xff, 0xcc, 0x66, 0x00));
|
2015-12-24 00:07:22 +01:00
|
|
|
|
|
|
|
// Load logo
|
|
|
|
Graphics::Surface *logo = bitmap.getSurface()->convertTo(g_system->getOverlayFormat(), bitmap.getPalette());
|
2016-06-17 12:47:00 +02:00
|
|
|
int lx = MAX((g_system->getOverlayWidth() - logo->w) / 2, 0);
|
|
|
|
int ly = MAX((g_system->getOverlayHeight() - logo->h) / 2, 0);
|
2015-12-24 00:07:22 +01:00
|
|
|
|
|
|
|
// Print version information
|
|
|
|
const Graphics::Font *font = FontMan.getFontByUsage(Graphics::FontManager::kConsoleFont);
|
|
|
|
int w = font->getStringWidth(gScummVMVersionDate);
|
|
|
|
int x = g_system->getOverlayWidth() - w - 5; // lx + logo->w - w + 5;
|
|
|
|
int y = g_system->getOverlayHeight() - font->getFontHeight() - 5; //ly + logo->h + 5;
|
|
|
|
font->drawString(&screen, gScummVMVersionDate, x, y, w, screen.format.ARGBToColor(0xff, 0, 0, 0));
|
|
|
|
|
2015-11-07 22:35:10 +01:00
|
|
|
g_system->copyRectToOverlay(screen.getPixels(), screen.pitch, 0, 0, screen.w, screen.h);
|
2015-12-28 10:21:48 +01:00
|
|
|
screen.free();
|
2015-11-07 22:35:10 +01:00
|
|
|
|
|
|
|
// Draw logo
|
2016-06-17 12:47:00 +02:00
|
|
|
int lw = MIN<uint16>(logo->w, g_system->getOverlayWidth() - lx);
|
|
|
|
int lh = MIN<uint16>(logo->h, g_system->getOverlayHeight() - ly);
|
|
|
|
|
|
|
|
g_system->copyRectToOverlay(logo->getPixels(), logo->pitch, lx, ly, lw, lh);
|
2015-12-28 10:21:48 +01:00
|
|
|
logo->free();
|
|
|
|
delete logo;
|
2015-11-07 22:35:10 +01:00
|
|
|
|
2018-06-23 14:49:50 +02:00
|
|
|
g_system->updateScreen();
|
|
|
|
|
2015-11-07 22:35:10 +01:00
|
|
|
// Delay 0.6 secs
|
|
|
|
uint time0 = g_system->getMillis();
|
|
|
|
Common::Event event;
|
2019-12-17 23:59:51 +01:00
|
|
|
|
|
|
|
// We must poll an event in order to have the window shown at least on Mac
|
2019-12-17 23:54:52 +01:00
|
|
|
g_system->getEventManager()->pollEvent(event);
|
|
|
|
|
2015-11-07 22:35:10 +01:00
|
|
|
while (time0 + 600 > g_system->getMillis()) {
|
|
|
|
g_system->delayMillis(10);
|
|
|
|
}
|
|
|
|
g_system->hideOverlay();
|
|
|
|
|
|
|
|
splash = true;
|
|
|
|
}
|
|
|
|
|
2017-10-01 16:23:22 -05:00
|
|
|
void initGraphicsModes(const Graphics::ModeList &modes) {
|
|
|
|
g_system->initSizeHint(modes);
|
|
|
|
}
|
|
|
|
|
2017-10-01 00:56:01 -05:00
|
|
|
void initGraphics(int width, int height, const Graphics::PixelFormat *format) {
|
2009-06-10 05:35:54 +00:00
|
|
|
|
2008-11-14 22:08:10 +00:00
|
|
|
g_system->beginGFXTransaction();
|
|
|
|
|
2017-10-01 00:56:01 -05:00
|
|
|
initCommonGFX();
|
2009-08-21 18:16:37 +00:00
|
|
|
#ifdef USE_RGB_COLOR
|
2009-07-10 10:43:48 +00:00
|
|
|
if (format)
|
|
|
|
g_system->initSize(width, height, format);
|
2010-01-25 01:39:44 +00:00
|
|
|
else {
|
2009-09-24 16:17:25 +00:00
|
|
|
Graphics::PixelFormat bestFormat = g_system->getSupportedFormats().front();
|
|
|
|
g_system->initSize(width, height, &bestFormat);
|
2009-07-10 10:43:48 +00:00
|
|
|
}
|
2009-06-23 02:02:51 +00:00
|
|
|
#else
|
2008-11-14 22:08:10 +00:00
|
|
|
g_system->initSize(width, height);
|
2009-06-23 02:02:51 +00:00
|
|
|
#endif
|
2008-11-14 22:08:10 +00:00
|
|
|
|
|
|
|
OSystem::TransactionError gfxError = g_system->endGFXTransaction();
|
|
|
|
|
2015-11-11 21:36:27 +01:00
|
|
|
if (!splash && !GUI::GuiManager::instance()._launched)
|
2015-11-07 22:35:10 +01:00
|
|
|
splashScreen();
|
|
|
|
|
2008-11-14 22:08:10 +00:00
|
|
|
if (gfxError == OSystem::kTransactionSuccess)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Error out on size switch failure
|
|
|
|
if (gfxError & OSystem::kTransactionSizeChangeFailed) {
|
2020-06-20 23:34:23 +05:30
|
|
|
Common::U32String message;
|
2020-08-20 11:45:58 +05:30
|
|
|
message = Common::U32String::format(_("Could not switch to resolution '%dx%d'."), width, height);
|
2008-11-14 22:08:10 +00:00
|
|
|
|
|
|
|
GUIErrorMessage(message);
|
2020-07-04 03:13:05 +05:30
|
|
|
error("Could not switch to resolution '%dx%d'.", width, height);
|
2008-11-14 22:08:10 +00:00
|
|
|
}
|
|
|
|
|
2009-06-15 09:45:19 +00:00
|
|
|
// Just show warnings then these occur:
|
2009-08-21 18:16:37 +00:00
|
|
|
#ifdef USE_RGB_COLOR
|
2009-06-30 07:30:57 +00:00
|
|
|
if (gfxError & OSystem::kTransactionFormatNotSupported) {
|
2020-06-13 22:12:25 +05:30
|
|
|
Common::U32String message = _("Could not initialize color format.");
|
2009-06-11 05:56:00 +00:00
|
|
|
|
2009-06-15 09:45:19 +00:00
|
|
|
GUI::MessageDialog dialog(message);
|
|
|
|
dialog.runModal();
|
2009-06-11 05:56:00 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2008-11-14 22:08:10 +00:00
|
|
|
if (gfxError & OSystem::kTransactionModeSwitchFailed) {
|
2020-06-20 23:34:23 +05:30
|
|
|
Common::U32String message;
|
2020-07-04 03:13:05 +05:30
|
|
|
message = Common::U32String::format(_("Could not switch to video mode '%s'."), ConfMan.get("gfx_mode").c_str());
|
2008-11-14 22:08:10 +00:00
|
|
|
|
|
|
|
GUI::MessageDialog dialog(message);
|
2009-01-01 15:06:43 +00:00
|
|
|
dialog.runModal();
|
2008-11-14 22:08:10 +00:00
|
|
|
}
|
2009-01-01 15:06:43 +00:00
|
|
|
|
2018-07-01 23:57:00 +01:00
|
|
|
if (gfxError & OSystem::kTransactionStretchModeSwitchFailed) {
|
2020-06-20 23:34:23 +05:30
|
|
|
Common::U32String message;
|
2020-07-04 03:13:05 +05:30
|
|
|
message = Common::U32String::format(_("Could not switch to stretch mode '%s'."), ConfMan.get("stretch_mode").c_str());
|
2018-07-01 23:57:00 +01:00
|
|
|
|
|
|
|
GUI::MessageDialog dialog(message);
|
|
|
|
dialog.runModal();
|
|
|
|
}
|
|
|
|
|
2008-11-14 22:08:10 +00:00
|
|
|
if (gfxError & OSystem::kTransactionAspectRatioFailed) {
|
2011-06-13 22:19:18 +01:00
|
|
|
GUI::MessageDialog dialog(_("Could not apply aspect ratio setting."));
|
2008-11-14 22:08:10 +00:00
|
|
|
dialog.runModal();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gfxError & OSystem::kTransactionFullscreenFailed) {
|
2011-06-13 22:19:18 +01:00
|
|
|
GUI::MessageDialog dialog(_("Could not apply fullscreen setting."));
|
2008-11-14 22:08:10 +00:00
|
|
|
dialog.runModal();
|
|
|
|
}
|
2016-10-12 22:32:36 +01:00
|
|
|
|
|
|
|
if (gfxError & OSystem::kTransactionFilteringFailed) {
|
|
|
|
GUI::MessageDialog dialog(_("Could not apply filtering setting."));
|
|
|
|
dialog.runModal();
|
|
|
|
}
|
2008-11-14 22:08:10 +00:00
|
|
|
}
|
2009-09-24 16:17:25 +00:00
|
|
|
|
2009-12-09 16:45:35 +00:00
|
|
|
/**
|
|
|
|
* Determines the first matching format between two lists.
|
|
|
|
*
|
|
|
|
* @param backend The higher priority list, meant to be a list of formats supported by the backend
|
|
|
|
* @param frontend The lower priority list, meant to be a list of formats supported by the engine
|
|
|
|
* @return The first item on the backend list that also occurs on the frontend list
|
|
|
|
* or PixelFormat::createFormatCLUT8() if no matching formats were found.
|
|
|
|
*/
|
2017-10-01 01:05:10 -05:00
|
|
|
inline Graphics::PixelFormat findCompatibleFormat(const Common::List<Graphics::PixelFormat> &backend, const Common::List<Graphics::PixelFormat> &frontend) {
|
2009-12-09 16:45:35 +00:00
|
|
|
#ifdef USE_RGB_COLOR
|
2017-10-01 01:05:10 -05:00
|
|
|
for (Common::List<Graphics::PixelFormat>::const_iterator i = backend.begin(); i != backend.end(); ++i) {
|
|
|
|
for (Common::List<Graphics::PixelFormat>::const_iterator j = frontend.begin(); j != frontend.end(); ++j) {
|
2009-12-09 16:45:35 +00:00
|
|
|
if (*i == *j)
|
|
|
|
return *i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2017-10-01 01:05:10 -05:00
|
|
|
return Graphics::PixelFormat::createFormatCLUT8();
|
2009-12-09 16:45:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-10-01 00:56:01 -05:00
|
|
|
void initGraphics(int width, int height, const Common::List<Graphics::PixelFormat> &formatList) {
|
2009-12-09 16:45:35 +00:00
|
|
|
Graphics::PixelFormat format = findCompatibleFormat(g_system->getSupportedFormats(), formatList);
|
2017-10-01 00:56:01 -05:00
|
|
|
initGraphics(width, height, &format);
|
2009-07-10 10:43:48 +00:00
|
|
|
}
|
2009-09-24 16:17:25 +00:00
|
|
|
|
2017-10-01 00:56:01 -05:00
|
|
|
void initGraphics(int width, int height) {
|
2009-07-10 10:43:48 +00:00
|
|
|
Graphics::PixelFormat format = Graphics::PixelFormat::createFormatCLUT8();
|
2017-10-01 00:56:01 -05:00
|
|
|
initGraphics(width, height, &format);
|
2009-07-10 10:43:48 +00:00
|
|
|
}
|
2008-11-14 22:08:10 +00:00
|
|
|
|
2020-10-14 06:49:28 +01:00
|
|
|
void initGraphics3d(int width, int height) {
|
2020-10-02 19:14:19 +02:00
|
|
|
g_system->beginGFXTransaction();
|
2020-10-13 23:00:19 +02:00
|
|
|
g_system->setGraphicsMode(0, OSystem::kGfxModeRender3d);
|
2020-10-02 19:14:19 +02:00
|
|
|
g_system->initSize(width, height);
|
2020-10-14 06:49:28 +01:00
|
|
|
g_system->setFeatureState(OSystem::kFeatureFullscreenMode, ConfMan.getBool("fullscreen")); // TODO: Replace this with initCommonGFX()
|
2020-10-16 20:17:29 +02:00
|
|
|
g_system->setFeatureState(OSystem::kFeatureAspectRatioCorrection, ConfMan.getBool("aspect_ratio")); // TODO: Replace this with initCommonGFX()
|
2020-10-02 19:14:19 +02:00
|
|
|
g_system->endGFXTransaction();
|
|
|
|
}
|
|
|
|
|
2020-06-20 23:34:23 +05:30
|
|
|
void GUIErrorMessageWithURL(const Common::U32String &msg, const char *url) {
|
2020-08-26 17:27:28 +02:00
|
|
|
GUIErrorMessage(msg, url);
|
|
|
|
}
|
|
|
|
|
2020-07-04 03:13:05 +05:30
|
|
|
void GUIErrorMessageWithURL(const Common::String &msg, const char *url) {
|
|
|
|
GUIErrorMessage(Common::U32String(msg), url);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GUIErrorMessage(const Common::String &msg, const char *url) {
|
|
|
|
GUIErrorMessage(Common::U32String(msg), url);
|
|
|
|
}
|
|
|
|
|
2020-06-20 23:34:23 +05:30
|
|
|
void GUIErrorMessage(const Common::U32String &msg, const char *url) {
|
2008-10-02 17:48:01 +00:00
|
|
|
g_system->setWindowCaption("Error");
|
|
|
|
g_system->beginGFXTransaction();
|
2017-10-01 00:56:01 -05:00
|
|
|
initCommonGFX();
|
2008-10-02 17:48:01 +00:00
|
|
|
g_system->initSize(320, 200);
|
2008-11-14 22:08:10 +00:00
|
|
|
if (g_system->endGFXTransaction() == OSystem::kTransactionSuccess) {
|
2020-08-26 17:27:28 +02:00
|
|
|
if (url) {
|
|
|
|
GUI::MessageDialogWithURL dialog(msg, url);
|
|
|
|
dialog.runModal();
|
|
|
|
} else {
|
|
|
|
GUI::MessageDialog dialog(msg);
|
|
|
|
dialog.runModal();
|
|
|
|
}
|
2008-11-14 22:08:10 +00:00
|
|
|
} else {
|
2020-06-20 23:34:23 +05:30
|
|
|
error("%s", msg.encode().c_str());
|
2008-11-14 22:08:10 +00:00
|
|
|
}
|
2004-11-24 00:14:21 +00:00
|
|
|
}
|
|
|
|
|
2018-12-19 06:31:26 +00:00
|
|
|
void GUIErrorMessageFormat(const char *fmt, ...) {
|
|
|
|
Common::String msg;
|
|
|
|
|
|
|
|
va_list va;
|
|
|
|
va_start(va, fmt);
|
|
|
|
msg = Common::String::vformat(fmt, va);
|
|
|
|
va_end(va);
|
|
|
|
|
|
|
|
GUIErrorMessage(msg);
|
|
|
|
}
|
|
|
|
|
2020-06-30 03:40:13 +05:30
|
|
|
void GUIErrorMessageFormat(Common::U32String fmt, ...) {
|
|
|
|
Common::U32String msg("");
|
|
|
|
|
|
|
|
va_list va;
|
|
|
|
va_start(va, fmt);
|
|
|
|
Common::U32String::vformat(fmt.begin(), fmt.end(), msg, va);
|
|
|
|
va_end(va);
|
|
|
|
|
|
|
|
GUIErrorMessage(msg);
|
|
|
|
}
|
|
|
|
|
2005-06-14 08:54:11 +00:00
|
|
|
void Engine::checkCD() {
|
2019-11-10 16:34:25 +00:00
|
|
|
#if defined(WIN32) && !defined(__SYMBIAN32__)
|
2005-06-14 08:54:11 +00:00
|
|
|
// 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.
|
|
|
|
|
2005-08-10 12:42:56 +00:00
|
|
|
#ifdef USE_VORBIS
|
2009-03-01 15:15:51 +00:00
|
|
|
if (Common::File::exists("track1.ogg") ||
|
|
|
|
Common::File::exists("track01.ogg"))
|
2005-06-14 08:54:11 +00:00
|
|
|
return;
|
|
|
|
#endif
|
|
|
|
#ifdef USE_FLAC
|
2009-03-01 15:15:51 +00:00
|
|
|
if (Common::File::exists("track1.fla") ||
|
|
|
|
Common::File::exists("track1.flac") ||
|
|
|
|
Common::File::exists("track01.fla") ||
|
|
|
|
Common::File::exists("track01.flac"))
|
2005-06-14 08:54:11 +00:00
|
|
|
return;
|
|
|
|
#endif
|
|
|
|
#ifdef USE_MAD
|
2009-03-01 15:15:51 +00:00
|
|
|
if (Common::File::exists("track1.mp3") ||
|
|
|
|
Common::File::exists("track01.mp3"))
|
2005-06-14 08:54:11 +00:00
|
|
|
return;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
char buffer[MAXPATHLEN];
|
|
|
|
int i;
|
|
|
|
|
2010-05-04 11:56:52 +00:00
|
|
|
const Common::FSNode gameDataDir(ConfMan.get("path"));
|
|
|
|
|
|
|
|
if (gameDataDir.getPath().empty()) {
|
2005-06-14 08:54:11 +00:00
|
|
|
// That's it! I give up!
|
|
|
|
if (getcwd(buffer, MAXPATHLEN) == NULL)
|
|
|
|
return;
|
|
|
|
} else
|
2010-05-05 17:54:12 +00:00
|
|
|
Common::strlcpy(buffer, gameDataDir.getPath().c_str(), sizeof(buffer));
|
2005-06-14 08:54:11 +00:00
|
|
|
|
|
|
|
for (i = 0; i < MAXPATHLEN - 1; i++) {
|
|
|
|
if (buffer[i] == '\\')
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
buffer[i + 1] = 0;
|
|
|
|
|
|
|
|
if (GetDriveType(buffer) == DRIVE_CDROM) {
|
|
|
|
GUI::MessageDialog dialog(
|
2011-06-13 22:19:18 +01:00
|
|
|
_("You appear to be playing this game directly\n"
|
2005-06-14 08:54:11 +00:00
|
|
|
"from the CD. This is known to cause problems,\n"
|
2008-09-30 12:27:38 +00:00
|
|
|
"and it is therefore recommended that you copy\n"
|
2005-06-14 08:54:11 +00:00
|
|
|
"the data files to your hard disk instead.\n"
|
2011-06-13 22:19:18 +01:00
|
|
|
"See the README file for details."), _("OK"));
|
2005-06-14 08:54:11 +00:00
|
|
|
dialog.runModal();
|
2008-10-26 19:27:26 +00:00
|
|
|
} 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(
|
2011-06-13 22:19:18 +01:00
|
|
|
_("This game has audio tracks in its disk. These\n"
|
2008-10-26 19:27:26 +00:00
|
|
|
"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"
|
2011-06-13 22:19:18 +01:00
|
|
|
"See the README file for details."), _("OK"));
|
2008-10-26 19:27:26 +00:00
|
|
|
dialog.runModal();
|
2005-06-14 08:54:11 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2020-02-04 22:25:13 -08:00
|
|
|
void Engine::handleAutoSave() {
|
|
|
|
const int diff = _system->getMillis() - _lastAutosaveTime;
|
|
|
|
|
|
|
|
if (_autosaveInterval != 0 && diff > (_autosaveInterval * 1000)) {
|
|
|
|
// Save the autosave
|
2020-02-05 08:01:10 -08:00
|
|
|
saveAutosaveIfEnabled();
|
2020-02-04 22:25:13 -08:00
|
|
|
}
|
2005-11-23 19:11:33 +00:00
|
|
|
}
|
|
|
|
|
2020-02-05 08:01:10 -08:00
|
|
|
void Engine::saveAutosaveIfEnabled() {
|
2020-02-09 16:21:27 -08:00
|
|
|
if (_autosaveInterval != 0) {
|
2020-02-09 19:08:23 -08:00
|
|
|
bool saveFlag = canSaveAutosaveCurrently();
|
|
|
|
|
|
|
|
if (saveFlag) {
|
|
|
|
// First check for an existing savegame in the slot, and if present, if it's an autosave
|
2020-10-02 13:10:18 +05:30
|
|
|
SaveStateDescriptor desc = getMetaEngine().querySaveMetaInfos(
|
2020-02-09 19:08:23 -08:00
|
|
|
_targetName.c_str(), getAutosaveSlot());
|
2020-02-10 18:55:31 -08:00
|
|
|
saveFlag = desc.getSaveSlot() == -1 || desc.isAutosave();
|
2020-02-09 19:08:23 -08:00
|
|
|
}
|
2020-02-09 16:21:27 -08:00
|
|
|
|
2020-06-13 22:12:25 +05:30
|
|
|
if (saveFlag && saveGameState(getAutosaveSlot(), Common::convertFromU32String(_("Autosave")), true).getCode() != Common::kNoError) {
|
2020-02-09 16:26:40 -08:00
|
|
|
// Couldn't autosave at the designated time
|
2020-02-09 19:08:23 -08:00
|
|
|
g_system->displayMessageOnOSD(_("Error occurred making autosave"));
|
|
|
|
saveFlag = false;
|
|
|
|
}
|
2020-02-09 16:26:40 -08:00
|
|
|
|
2020-02-09 19:08:23 -08:00
|
|
|
if (!saveFlag) {
|
2020-02-09 16:26:40 -08:00
|
|
|
// Set the next autosave interval to be in 5 minutes, rather than whatever
|
|
|
|
// full autosave interval the user has selected
|
2020-02-09 16:21:27 -08:00
|
|
|
_lastAutosaveTime = _system->getMillis() + (5 * 60 * 1000) - _autosaveInterval;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2020-02-05 08:01:10 -08:00
|
|
|
|
|
|
|
// Reset the last autosave time
|
|
|
|
_lastAutosaveTime = _system->getMillis();
|
|
|
|
}
|
|
|
|
|
2008-11-15 03:16:41 +00:00
|
|
|
void Engine::errorString(const char *buf1, char *buf2, int size) {
|
2010-05-05 17:54:12 +00:00
|
|
|
Common::strlcpy(buf2, buf1, size);
|
2006-02-12 01:23:58 +00:00
|
|
|
}
|
|
|
|
|
2020-03-15 13:55:54 +01:00
|
|
|
PauseToken Engine::pauseEngine() {
|
|
|
|
assert(_pauseLevel >= 0);
|
2007-06-30 22:22:25 +00:00
|
|
|
|
2020-03-15 13:55:54 +01:00
|
|
|
_pauseLevel++;
|
2007-06-30 22:22:25 +00:00
|
|
|
|
2020-03-15 13:55:54 +01:00
|
|
|
if (_pauseLevel == 1) {
|
2010-10-27 22:37:51 +00:00
|
|
|
_pauseStartTime = _system->getMillis();
|
2007-06-30 22:22:25 +00:00
|
|
|
pauseEngineIntern(true);
|
2020-03-15 13:55:54 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return PauseToken(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Engine::resumeEngine() {
|
|
|
|
assert(_pauseLevel > 0);
|
|
|
|
|
|
|
|
_pauseLevel--;
|
|
|
|
|
|
|
|
if (_pauseLevel == 0) {
|
2007-06-30 22:22:25 +00:00
|
|
|
pauseEngineIntern(false);
|
2010-10-27 22:37:51 +00:00
|
|
|
_engineStartTime += _system->getMillis() - _pauseStartTime;
|
|
|
|
_pauseStartTime = 0;
|
2007-06-30 22:22:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Engine::pauseEngineIntern(bool pause) {
|
|
|
|
// By default, just (un)pause all digital sounds
|
|
|
|
_mixer->pauseAll(pause);
|
|
|
|
}
|
2008-06-24 21:15:30 +00:00
|
|
|
|
2008-10-02 17:48:01 +00:00
|
|
|
void Engine::openMainMenuDialog() {
|
2008-06-24 21:15:30 +00:00
|
|
|
if (!_mainMenuDialog)
|
|
|
|
_mainMenuDialog = new MainMenuDialog(this);
|
2019-07-22 19:42:30 +02:00
|
|
|
#ifdef USE_TTS
|
|
|
|
Common::TextToSpeechManager *ttsMan = g_system->getTextToSpeechManager();
|
2020-05-25 18:29:41 +01:00
|
|
|
if (ttsMan != nullptr) {
|
|
|
|
ttsMan->pushState();
|
|
|
|
g_gui.initTextToSpeech();
|
|
|
|
}
|
2019-07-22 19:42:30 +02:00
|
|
|
#endif
|
2011-06-25 15:41:58 +02:00
|
|
|
|
|
|
|
setGameToLoadSlot(-1);
|
|
|
|
|
2020-08-27 22:37:50 +01:00
|
|
|
bool hasVKeyb = g_system->getFeatureState(OSystem::kFeatureVirtualKeyboard);
|
|
|
|
if (hasVKeyb)
|
|
|
|
g_system->setFeatureState(OSystem::kFeatureVirtualKeyboard, false);
|
|
|
|
|
2008-06-24 21:15:30 +00:00
|
|
|
runDialog(*_mainMenuDialog);
|
2011-06-25 15:41:58 +02:00
|
|
|
|
2020-08-27 22:37:50 +01:00
|
|
|
if (hasVKeyb)
|
|
|
|
g_system->setFeatureState(OSystem::kFeatureVirtualKeyboard, true);
|
|
|
|
|
2011-06-25 15:41:58 +02:00
|
|
|
// Load savegame after main menu execution
|
|
|
|
// (not from inside the menu loop to avoid
|
2019-09-13 09:29:25 +02:00
|
|
|
// mouse cursor glitches and similar bugs,
|
2011-06-25 15:41:58 +02:00
|
|
|
// e.g. #2822778).
|
2012-02-21 22:30:29 +00:00
|
|
|
if (_saveSlotToLoad >= 0) {
|
|
|
|
Common::Error status = loadGameState(_saveSlotToLoad);
|
|
|
|
if (status.getCode() != Common::kNoError) {
|
2020-06-20 23:34:23 +05:30
|
|
|
Common::U32String failMessage = Common::U32String::format(_("Failed to load saved game (%s)! "
|
2012-02-21 22:30:29 +00:00
|
|
|
"Please consult the README for basic information, and for "
|
2020-06-20 23:34:23 +05:30
|
|
|
"instructions on how to obtain further assistance."), status.getDesc().c_str());
|
2012-02-21 22:30:29 +00:00
|
|
|
GUI::MessageDialog dialog(failMessage);
|
|
|
|
dialog.runModal();
|
|
|
|
}
|
|
|
|
}
|
2011-06-25 15:41:58 +02:00
|
|
|
|
2019-07-22 19:42:30 +02:00
|
|
|
#ifdef USE_TTS
|
2020-05-25 18:29:41 +01:00
|
|
|
if (ttsMan != nullptr)
|
|
|
|
ttsMan->popState();
|
2019-07-22 19:42:30 +02:00
|
|
|
#endif
|
2020-06-22 00:15:22 +01:00
|
|
|
|
2020-05-11 12:05:56 +01:00
|
|
|
g_system->applyBackendSettings();
|
2020-06-22 00:15:22 +01:00
|
|
|
applyGameSettings();
|
|
|
|
syncSoundSettings();
|
2008-06-24 21:15:30 +00:00
|
|
|
}
|
|
|
|
|
2011-04-25 15:26:38 -05:00
|
|
|
bool Engine::warnUserAboutUnsupportedGame() {
|
|
|
|
if (ConfMan.getBool("enable_unsupported_game_warning")) {
|
2011-06-20 23:12:26 +01:00
|
|
|
GUI::MessageDialog alert(_("WARNING: The game you are about to start is"
|
2011-04-25 15:26:38 -05:00
|
|
|
" not yet fully supported by ScummVM. As such, it is likely to be"
|
2016-09-18 16:39:11 +01:00
|
|
|
" unstable, and any saved game you make might not work in future"
|
2011-06-20 23:12:26 +01:00
|
|
|
" versions of ScummVM."), _("Start anyway"), _("Cancel"));
|
2011-04-25 15:26:38 -05:00
|
|
|
return alert.runModal() == GUI::kMessageOK;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-11-08 20:05:35 +01:00
|
|
|
void Engine::errorUnsupportedGame(Common::String &extraMsg) {
|
|
|
|
Common::String message = extraMsg.empty() ? _("This game is not supported.") : _("This game is not supported for the following reason:\n\n");
|
|
|
|
message += _(extraMsg);
|
|
|
|
message += "\n\n";
|
|
|
|
GUI::MessageDialog(message).runModal();
|
|
|
|
}
|
|
|
|
|
2010-10-27 22:37:51 +00:00
|
|
|
uint32 Engine::getTotalPlayTime() const {
|
|
|
|
if (!_pauseLevel)
|
|
|
|
return _system->getMillis() - _engineStartTime;
|
|
|
|
else
|
|
|
|
return _pauseStartTime - _engineStartTime;
|
|
|
|
}
|
|
|
|
|
2010-10-27 22:52:02 +00:00
|
|
|
void Engine::setTotalPlayTime(uint32 time) {
|
2010-10-27 22:37:51 +00:00
|
|
|
const uint32 currentTime = _system->getMillis();
|
|
|
|
|
|
|
|
// We need to reset the pause start time here in case the engine is already
|
|
|
|
// paused to avoid any incorrect play time counting.
|
|
|
|
if (_pauseLevel > 0)
|
|
|
|
_pauseStartTime = currentTime;
|
|
|
|
|
|
|
|
_engineStartTime = currentTime - time;
|
|
|
|
}
|
|
|
|
|
2008-10-02 17:48:01 +00:00
|
|
|
int Engine::runDialog(GUI::Dialog &dialog) {
|
2020-03-15 13:55:54 +01:00
|
|
|
PauseToken pt = pauseEngine();
|
2008-08-06 21:47:20 +00:00
|
|
|
int result = dialog.runModal();
|
2008-06-24 21:15:30 +00:00
|
|
|
|
2008-08-06 21:47:20 +00:00
|
|
|
return result;
|
2008-06-24 21:15:30 +00:00
|
|
|
}
|
|
|
|
|
2011-06-25 15:41:58 +02:00
|
|
|
void Engine::setGameToLoadSlot(int slot) {
|
|
|
|
_saveSlotToLoad = slot;
|
|
|
|
}
|
|
|
|
|
2008-06-26 23:22:28 +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-06 17:36:06 +00:00
|
|
|
bool mute = false;
|
|
|
|
if (ConfMan.hasKey("mute"))
|
|
|
|
mute = ConfMan.getBool("mute");
|
|
|
|
|
2011-11-13 22:25:19 +01:00
|
|
|
// We need to handle the speech mute separately here. This is because the
|
|
|
|
// engine code should be able to rely on all speech sounds muted when the
|
|
|
|
// user specified subtitles only mode, which results in "speech_mute" to
|
|
|
|
// be set to "true". The global mute setting has precedence over the
|
|
|
|
// speech mute setting though.
|
|
|
|
bool speechMute = mute;
|
|
|
|
if (!speechMute)
|
|
|
|
speechMute = ConfMan.getBool("speech_mute");
|
|
|
|
|
2011-04-16 18:20:37 +02:00
|
|
|
_mixer->muteSoundType(Audio::Mixer::kPlainSoundType, mute);
|
|
|
|
_mixer->muteSoundType(Audio::Mixer::kMusicSoundType, mute);
|
|
|
|
_mixer->muteSoundType(Audio::Mixer::kSFXSoundType, mute);
|
2011-11-13 22:25:19 +01:00
|
|
|
_mixer->muteSoundType(Audio::Mixer::kSpeechSoundType, speechMute);
|
|
|
|
|
2011-03-20 23:45:52 +01:00
|
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kPlainSoundType, Audio::Mixer::kMaxMixerVolume);
|
|
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kMusicSoundType, soundVolumeMusic);
|
|
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kSFXSoundType, soundVolumeSFX);
|
|
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kSpeechSoundType, soundVolumeSpeech);
|
2009-06-06 17:36:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Engine::flipMute() {
|
2009-09-24 16:17:52 +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-06 17:36:06 +00:00
|
|
|
|
|
|
|
if (ConfMan.hasKey("mute")) {
|
|
|
|
mute = !ConfMan.getBool("mute");
|
|
|
|
}
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-06-06 17:36:06 +00:00
|
|
|
ConfMan.setBool("mute", mute);
|
|
|
|
|
|
|
|
syncSoundSettings();
|
2008-06-26 23:22:28 +00:00
|
|
|
}
|
2008-07-09 02:27:05 +00:00
|
|
|
|
2008-11-06 17:05:54 +00:00
|
|
|
Common::Error Engine::loadGameState(int slot) {
|
2020-02-05 08:01:10 -08:00
|
|
|
// In case autosaves are on, do a save first before loading the new save
|
|
|
|
saveAutosaveIfEnabled();
|
|
|
|
|
2020-02-02 11:11:41 -08:00
|
|
|
Common::InSaveFile *saveFile = _saveFileMan->openForLoading(getSaveStateName(slot));
|
|
|
|
|
|
|
|
if (!saveFile)
|
|
|
|
return Common::kReadingFailed;
|
|
|
|
|
|
|
|
Common::Error result = loadGameStream(saveFile);
|
|
|
|
if (result.getCode() == Common::kNoError) {
|
|
|
|
ExtendedSavegameHeader header;
|
2020-10-02 11:16:09 +05:30
|
|
|
if (MetaEngine::readSavegameHeader(saveFile, &header))
|
2020-02-02 11:11:41 -08:00
|
|
|
setTotalPlayTime(header.playtime);
|
|
|
|
}
|
|
|
|
|
|
|
|
delete saveFile;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
Common::Error Engine::loadGameStream(Common::SeekableReadStream *stream) {
|
|
|
|
// Default to returning an error when not implemented
|
|
|
|
return Common::kReadingFailed;
|
2008-11-03 18:32:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Engine::canLoadGameStateCurrently() {
|
|
|
|
// Do not allow loading by default
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-02-02 17:51:13 -08:00
|
|
|
Common::Error Engine::saveGameState(int slot, const Common::String &desc, bool isAutosave) {
|
2020-02-02 11:11:41 -08:00
|
|
|
Common::OutSaveFile *saveFile = _saveFileMan->openForSaving(getSaveStateName(slot));
|
|
|
|
|
|
|
|
if (!saveFile)
|
|
|
|
return Common::kWritingFailed;
|
|
|
|
|
2020-02-02 17:51:13 -08:00
|
|
|
Common::Error result = saveGameStream(saveFile, isAutosave);
|
2020-02-02 11:11:41 -08:00
|
|
|
if (result.getCode() == Common::kNoError) {
|
2020-10-02 11:16:09 +05:30
|
|
|
MetaEngine::appendExtendedSave(saveFile, getTotalPlayTime() / 1000, desc, isAutosave);
|
2020-02-02 11:11:41 -08:00
|
|
|
|
|
|
|
saveFile->finalize();
|
|
|
|
}
|
|
|
|
|
|
|
|
delete saveFile;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2020-02-02 17:51:13 -08:00
|
|
|
Common::Error Engine::saveGameStream(Common::WriteStream *stream, bool isAutosave) {
|
2020-02-02 11:11:41 -08:00
|
|
|
// Default to returning an error when not implemented
|
|
|
|
return Common::kWritingFailed;
|
2008-11-03 18:32:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Engine::canSaveGameStateCurrently() {
|
|
|
|
// Do not allow saving by default
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-02-02 16:19:29 -08:00
|
|
|
bool Engine::loadGameDialog() {
|
2020-02-03 18:34:33 -08:00
|
|
|
if (!canLoadGameStateCurrently()) {
|
2020-02-04 18:43:25 -08:00
|
|
|
g_system->displayMessageOnOSD(_("Loading game is currently unavailable"));
|
2020-02-02 16:19:29 -08:00
|
|
|
return false;
|
2020-02-03 18:34:33 -08:00
|
|
|
}
|
2020-02-02 16:19:29 -08:00
|
|
|
|
2020-07-01 23:31:48 +05:30
|
|
|
GUI::SaveLoadChooser *dialog = new GUI::SaveLoadChooser(_("Load game:"), _("Load"), false);
|
2020-03-15 13:55:54 +01:00
|
|
|
|
|
|
|
int slotNum;
|
|
|
|
{
|
|
|
|
PauseToken pt = pauseEngine();
|
|
|
|
slotNum = dialog->runModalWithCurrentTarget();
|
|
|
|
}
|
|
|
|
|
2020-02-02 16:19:29 -08:00
|
|
|
delete dialog;
|
|
|
|
|
2020-02-05 14:25:56 +01:00
|
|
|
if (slotNum < 0)
|
|
|
|
return false;
|
2020-02-02 16:19:29 -08:00
|
|
|
|
2020-02-05 14:25:56 +01:00
|
|
|
Common::Error loadError = loadGameState(slotNum);
|
|
|
|
if (loadError.getCode() != Common::kNoError) {
|
|
|
|
GUI::MessageDialog errorDialog(loadError.getDesc());
|
|
|
|
errorDialog.runModal();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2020-02-02 16:19:29 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Engine::saveGameDialog() {
|
2020-02-03 18:34:33 -08:00
|
|
|
if (!canSaveGameStateCurrently()) {
|
2020-02-04 18:43:25 -08:00
|
|
|
g_system->displayMessageOnOSD(_("Saving game is currently unavailable"));
|
2020-02-02 16:19:29 -08:00
|
|
|
return false;
|
2020-02-03 18:34:33 -08:00
|
|
|
}
|
2020-02-02 16:19:29 -08:00
|
|
|
|
2020-07-01 23:31:48 +05:30
|
|
|
GUI::SaveLoadChooser *dialog = new GUI::SaveLoadChooser(_("Save game:"), _("Save"), true);
|
2020-03-15 13:55:54 +01:00
|
|
|
int slotNum;
|
|
|
|
{
|
|
|
|
PauseToken pt = pauseEngine();
|
|
|
|
slotNum = dialog->runModalWithCurrentTarget();
|
|
|
|
}
|
2020-02-02 16:19:29 -08:00
|
|
|
|
2020-08-29 19:08:39 +05:30
|
|
|
Common::String desc = dialog->getResultString();
|
2020-02-02 16:19:29 -08:00
|
|
|
if (desc.empty())
|
|
|
|
desc = dialog->createDefaultSaveDescription(slotNum);
|
|
|
|
|
|
|
|
delete dialog;
|
|
|
|
|
2020-02-05 14:25:56 +01:00
|
|
|
if (slotNum < 0)
|
|
|
|
return false;
|
2020-02-02 16:19:29 -08:00
|
|
|
|
2020-08-29 19:08:39 +05:30
|
|
|
Common::Error saveError = saveGameState(slotNum, desc);
|
2020-02-05 14:25:56 +01:00
|
|
|
if (saveError.getCode() != Common::kNoError) {
|
|
|
|
GUI::MessageDialog errorDialog(saveError.getDesc());
|
|
|
|
errorDialog.runModal();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2020-02-02 16:19:29 -08:00
|
|
|
}
|
|
|
|
|
2008-07-09 02:27:05 +00:00
|
|
|
void Engine::quitGame() {
|
|
|
|
Common::Event event;
|
|
|
|
|
|
|
|
event.type = Common::EVENT_QUIT;
|
2009-01-30 16:04:39 +00:00
|
|
|
g_system->getEventManager()->pushEvent(event);
|
2008-07-09 02:27:05 +00:00
|
|
|
}
|
2008-08-16 02:53:16 +00:00
|
|
|
|
2009-01-30 16:04:39 +00:00
|
|
|
bool Engine::shouldQuit() {
|
|
|
|
Common::EventManager *eventMan = g_system->getEventManager();
|
2020-05-11 08:47:44 -04:00
|
|
|
return (eventMan->shouldQuit() || eventMan->shouldReturnToLauncher());
|
2008-10-06 12:48:52 +00:00
|
|
|
}
|
|
|
|
|
2020-02-09 11:06:20 -08:00
|
|
|
GUI::Debugger *Engine::getOrCreateDebugger() {
|
|
|
|
if (!_debugger)
|
|
|
|
// Create a bare-bones debugger. This is useful for engines without their own
|
|
|
|
// debugger when an error occurs
|
|
|
|
_debugger = new GUI::Debugger();
|
|
|
|
|
|
|
|
return _debugger;
|
|
|
|
}
|
|
|
|
|
2008-11-04 16:11:40 +00:00
|
|
|
/*
|
|
|
|
EnginePlugin *Engine::getMetaEnginePlugin() const {
|
2016-09-15 18:39:45 +02:00
|
|
|
return EngineMan.findPlugin(ConfMan.get("engineid"));
|
2008-08-16 02:53:16 +00:00
|
|
|
}
|
|
|
|
|
2008-11-04 16:11:40 +00:00
|
|
|
*/
|
2020-02-09 19:08:23 -08:00
|
|
|
|
2020-10-11 23:14:39 +02:00
|
|
|
MetaEngineDetection &Engine::getMetaEngineDetection() {
|
2020-02-09 19:08:23 -08:00
|
|
|
const Plugin *plugin = EngineMan.findPlugin(ConfMan.get("engineid"));
|
|
|
|
assert(plugin);
|
2020-10-11 23:14:39 +02:00
|
|
|
return plugin->get<MetaEngineDetection>();
|
2020-02-09 19:08:23 -08:00
|
|
|
}
|
2020-03-15 13:55:54 +01:00
|
|
|
|
2020-10-02 13:10:18 +05:30
|
|
|
MetaEngine &Engine::getMetaEngine() {
|
2020-08-03 00:58:30 +05:30
|
|
|
const Plugin *metaEnginePlugin = EngineMan.findPlugin(ConfMan.get("engineid"));
|
|
|
|
assert(metaEnginePlugin);
|
|
|
|
|
2020-09-30 14:05:56 +05:30
|
|
|
const Plugin *enginePlugin = PluginMan.getEngineFromMetaEngine(metaEnginePlugin);
|
2020-08-03 00:58:30 +05:30
|
|
|
assert(enginePlugin);
|
|
|
|
|
2020-10-02 11:16:09 +05:30
|
|
|
return enginePlugin->get<MetaEngine>();
|
2020-08-03 00:58:30 +05:30
|
|
|
}
|
|
|
|
|
2020-03-15 13:55:54 +01:00
|
|
|
PauseToken::PauseToken() : _engine(nullptr) {}
|
|
|
|
|
|
|
|
PauseToken::PauseToken(Engine *engine) : _engine(engine) {}
|
|
|
|
|
|
|
|
void PauseToken::operator=(const PauseToken &t2) {
|
|
|
|
if (_engine) {
|
2020-05-10 23:11:46 +02:00
|
|
|
error("Tried to assign to an already busy PauseToken");
|
2020-03-15 13:55:54 +01:00
|
|
|
}
|
|
|
|
_engine = t2._engine;
|
|
|
|
if (_engine) {
|
|
|
|
_engine->_pauseLevel++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
PauseToken::PauseToken(const PauseToken &t2) : _engine(t2._engine) {
|
|
|
|
if (_engine) {
|
|
|
|
_engine->_pauseLevel++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-15 14:23:23 +01:00
|
|
|
void PauseToken::clear() {
|
2020-05-10 23:15:55 +02:00
|
|
|
if (!_engine) {
|
|
|
|
error("Tried to clear an already cleared PauseToken");
|
2020-03-15 14:23:23 +01:00
|
|
|
}
|
2020-05-10 23:15:55 +02:00
|
|
|
_engine->resumeEngine();
|
2020-03-15 14:23:23 +01:00
|
|
|
_engine = nullptr;
|
|
|
|
}
|
|
|
|
|
2020-03-15 13:55:54 +01:00
|
|
|
PauseToken::~PauseToken() {
|
|
|
|
if (_engine) {
|
|
|
|
_engine->resumeEngine();
|
|
|
|
}
|
|
|
|
}
|
2020-03-15 16:35:59 +01:00
|
|
|
|
|
|
|
#if __cplusplus >= 201103L
|
|
|
|
PauseToken::PauseToken(PauseToken &&t2) : _engine(t2._engine) {
|
|
|
|
t2._engine = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PauseToken::operator=(PauseToken &&t2) {
|
|
|
|
if (_engine) {
|
|
|
|
error("Tried to assign to an already busy PauseToken");
|
|
|
|
}
|
|
|
|
_engine = t2._engine;
|
|
|
|
t2._engine = nullptr;
|
|
|
|
}
|
|
|
|
#endif
|