scummvm/engines/stark/stark.cpp

498 lines
16 KiB
C++
Raw Normal View History

2014-09-21 18:19:07 +02:00
/* ResidualVM - A 3D game interpreter
*
2014-09-21 18:19:07 +02:00
* ResidualVM 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.
*
2014-09-21 18:19:07 +02: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
2014-09-21 18:19:07 +02:00
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
2014-09-21 18:19:07 +02: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 "engines/stark/stark.h"
#include "engines/stark/console.h"
#include "engines/stark/debug.h"
#include "engines/stark/resources/level.h"
#include "engines/stark/resources/location.h"
#include "engines/stark/savemetadata.h"
2014-09-21 17:26:20 +02:00
#include "engines/stark/scene.h"
#include "engines/stark/services/userinterface.h"
#include "engines/stark/services/archiveloader.h"
2015-01-11 15:25:43 +01:00
#include "engines/stark/services/dialogplayer.h"
#include "engines/stark/services/diary.h"
#include "engines/stark/services/fontprovider.h"
#include "engines/stark/services/gameinterface.h"
#include "engines/stark/services/global.h"
#include "engines/stark/services/resourceprovider.h"
#include "engines/stark/services/services.h"
#include "engines/stark/services/stateprovider.h"
2015-02-22 11:26:31 +01:00
#include "engines/stark/services/staticprovider.h"
#include "engines/stark/services/settings.h"
#include "engines/stark/services/gamechapter.h"
2018-06-30 13:48:55 +08:00
#include "engines/stark/services/gamemessage.h"
2014-09-21 17:26:20 +02:00
#include "engines/stark/gfx/driver.h"
#include "engines/stark/gfx/framelimiter.h"
#include "common/config-manager.h"
2015-07-21 08:21:33 +02:00
#include "common/debug-channels.h"
#include "common/events.h"
2015-01-02 10:33:42 +01:00
#include "common/random.h"
2014-12-29 23:11:22 +01:00
#include "common/savefile.h"
2011-09-14 02:09:51 +02:00
#include "common/system.h"
#include "audio/mixer.h"
namespace Stark {
StarkEngine::StarkEngine(OSystem *syst, const ADGameDescription *gameDesc) :
Engine(syst),
_gameDescription(gameDesc),
_gfx(nullptr),
_frameLimiter(nullptr),
_scene(nullptr),
_console(nullptr),
_global(nullptr),
_gameInterface(nullptr),
_archiveLoader(nullptr),
2014-12-28 18:22:25 +01:00
_stateProvider(nullptr),
2015-02-22 11:26:31 +01:00
_staticProvider(nullptr),
2015-01-02 10:33:42 +01:00
_resourceProvider(nullptr),
2015-01-11 15:25:43 +01:00
_randomSource(nullptr),
_dialogPlayer(nullptr),
_diary(nullptr),
2015-08-15 13:55:00 +02:00
_userInterface(nullptr),
_fontProvider(nullptr),
_settings(nullptr),
_gameChapter(nullptr),
2018-06-30 13:48:55 +08:00
_gameMessage(nullptr),
2015-08-15 13:55:00 +02:00
_lastClickTime(0) {
// Add the available debug channels
2011-09-14 02:09:51 +02:00
DebugMan.addDebugChannel(kDebugArchive, "Archive", "Debug the archive loading");
DebugMan.addDebugChannel(kDebugXMG, "XMG", "Debug the loading of XMG images");
DebugMan.addDebugChannel(kDebugXRC, "XRC", "Debug the loading of XRC resource trees");
DebugMan.addDebugChannel(kDebugUnknown, "Unknown", "Debug unknown values on the data");
}
StarkEngine::~StarkEngine() {
delete _gameInterface;
delete _diary;
2015-01-11 15:25:43 +01:00
delete _dialogPlayer;
2015-01-02 10:33:42 +01:00
delete _randomSource;
delete _scene;
delete _console;
delete _gfx;
delete _frameLimiter;
2015-02-22 11:26:31 +01:00
delete _staticProvider;
delete _resourceProvider;
delete _global;
2014-12-28 18:22:25 +01:00
delete _stateProvider;
delete _archiveLoader;
delete _userInterface;
delete _fontProvider;
delete _settings;
delete _gameChapter;
2018-06-30 13:48:55 +08:00
delete _gameMessage;
StarkServices::destroy();
}
Common::Error StarkEngine::run() {
_console = new Console();
_gfx = Gfx::Driver::create();
2010-01-21 20:29:34 +10:30
2010-01-21 21:26:06 +10:30
// Get the screen prepared
_gfx->init();
2010-01-21 20:29:34 +10:30
_frameLimiter = new Gfx::FrameLimiter(_system, ConfMan.getInt("engine_speed"));
_archiveLoader = new ArchiveLoader();
2014-12-28 18:22:25 +01:00
_stateProvider = new StateProvider();
_global = new Global();
2014-12-28 18:22:25 +01:00
_resourceProvider = new ResourceProvider(_archiveLoader, _stateProvider, _global);
_staticProvider = new StaticProvider(_archiveLoader);
2015-01-02 10:33:42 +01:00
_randomSource = new Common::RandomSource("stark");
_fontProvider = new FontProvider();
_scene = new Scene(_gfx);
2015-01-11 15:25:43 +01:00
_dialogPlayer = new DialogPlayer();
_diary = new Diary();
_gameInterface = new GameInterface();
_userInterface = new UserInterface(_gfx);
_settings = new Settings(_mixer, _gameDescription);
_gameChapter = new GameChapter();
2018-06-30 13:48:55 +08:00
_gameMessage = new GameMessage();
// Setup the public services
StarkServices &services = StarkServices::instance();
services.archiveLoader = _archiveLoader;
2015-01-11 15:25:43 +01:00
services.dialogPlayer = _dialogPlayer;
services.diary = _diary;
2015-02-13 17:07:24 +01:00
services.gfx = _gfx;
services.global = _global;
2015-01-11 15:25:43 +01:00
services.resourceProvider = _resourceProvider;
2015-01-02 10:33:42 +01:00
services.randomSource = _randomSource;
services.scene = _scene;
2015-02-22 11:26:31 +01:00
services.staticProvider = _staticProvider;
services.gameInterface = _gameInterface;
services.userInterface = _userInterface;
services.fontProvider = _fontProvider;
services.settings = _settings;
services.gameChapter = _gameChapter;
2018-06-30 13:48:55 +08:00
services.gameMessage = _gameMessage;
// Load global resources
2015-02-22 11:26:31 +01:00
_staticProvider->init();
_fontProvider->initFonts();
// Initialize the UI
_userInterface->init();
// Load through ResidualVM launcher
if (ConfMan.hasKey("save_slot")) {
loadGameState(ConfMan.getInt("save_slot"));
}
2010-01-21 21:26:06 +10:30
// Start running
mainLoop();
2015-02-22 11:26:31 +01:00
_staticProvider->shutdown();
_resourceProvider->shutdown();
return Common::kNoError;
}
void StarkEngine::mainLoop() {
while (!shouldQuit()) {
_frameLimiter->startFrame();
2015-02-24 22:54:03 +01:00
processEvents();
if (_userInterface->shouldExit()) {
2015-02-24 22:54:03 +01:00
quitGame();
break;
}
2018-06-30 11:53:48 +08:00
if (_userInterface->hasQuitToMainMenuRequest()) {
_userInterface->performQuitToMainMenu();
}
if (_resourceProvider->hasLocationChangeRequest()) {
_global->setNormalSpeed();
_resourceProvider->performLocationChange();
}
updateDisplayScene();
// Swap buffers
_frameLimiter->delayBeforeSwap();
_gfx->flipBuffer();
2010-01-21 20:29:34 +10:30
}
}
void StarkEngine::processEvents() {
Common::Event e;
while (g_system->getEventManager()->pollEvent(e)) {
// Handle any buttons, keys and joystick operations
2018-07-02 16:32:54 +08:00
if (isPaused()) {
// Only pressing key P to resume the game is allowed when the game is paused
if (e.type == Common::EVENT_KEYDOWN && e.kbd.keycode == Common::KEYCODE_p) {
pauseEngine(false);
}
continue;
}
if (e.type == Common::EVENT_KEYDOWN) {
if (e.kbd.keycode == Common::KEYCODE_d) {
if (e.kbd.flags & Common::KBD_CTRL) {
_console->attach();
_console->onFrame();
}
} else if (e.kbd.keycode == Common::KEYCODE_ESCAPE) {
// Quick-hack for now.
bool skipped = _gameInterface->skipCurrentSpeeches();
if (!skipped) {
skipped = _userInterface->skipFMV();
}
2018-05-24 15:56:04 +08:00
if (!skipped && StarkSettings->getBoolSetting(Settings::kTimeSkip)) {
_global->setFastForward();
}
2017-08-26 22:25:18 +02:00
} else if ((e.kbd.keycode == Common::KEYCODE_RETURN
|| e.kbd.keycode == Common::KEYCODE_KP_ENTER)
&& e.kbd.hasFlags(Common::KBD_ALT)) {
_gfx->toggleFullscreen();
2018-07-01 14:21:01 +08:00
} else if (e.kbd.keycode == Common::KEYCODE_F1) {
_userInterface->toggleScreen(Screen::kScreenDiaryIndex);
} else if (e.kbd.keycode == Common::KEYCODE_F2) {
_userInterface->toggleScreen(Screen::kScreenSaveMenu);
} else if (e.kbd.keycode == Common::KEYCODE_F3) {
_userInterface->toggleScreen(Screen::kScreenLoadMenu);
} else if (e.kbd.keycode == Common::KEYCODE_F4) {
_userInterface->toggleScreen(Screen::kScreenDialog);
} else if (e.kbd.keycode == Common::KEYCODE_F5) {
if (_diary->isEnabled()) {
_userInterface->toggleScreen(Screen::kScreenDiaryPages);
}
} else if (e.kbd.keycode == Common::KEYCODE_F6) {
_userInterface->toggleScreen(Screen::kScreenFMVMenu);
} else if (e.kbd.keycode == Common::KEYCODE_F7) {
_userInterface->toggleScreen(Screen::kScreenSettingsMenu);
2018-07-02 11:56:05 +08:00
} else if (e.kbd.keycode == Common::KEYCODE_F9) {
_userInterface->requestToggleSubtitle();
} else if (e.kbd.keycode == Common::KEYCODE_F10) {
if (_userInterface->isInGameScreen() || _userInterface->isInDiaryIndexScreen()) {
if (_userInterface->confirm(GameMessage::kQuitGamePrompt)) {
_userInterface->requestQuitToMainMenu();
}
}
} else if (e.kbd.keycode == Common::KEYCODE_a) {
if (_userInterface->isInGameScreen()) {
_userInterface->cycleBackInventory();
}
} else if (e.kbd.keycode == Common::KEYCODE_s) {
if (_userInterface->isInGameScreen()) {
_userInterface->cycleForwardInventory();
}
2018-07-01 15:08:13 +08:00
} else if (e.kbd.keycode == Common::KEYCODE_i) {
if (_userInterface->isInGameScreen()) {
_userInterface->inventoryOpen(!_userInterface->isInventoryOpen());
}
2018-07-01 15:07:48 +08:00
} else if ((e.kbd.keycode == Common::KEYCODE_x
|| e.kbd.keycode == Common::KEYCODE_q)
&& e.kbd.hasFlags(Common::KBD_ALT)) {
if (_userInterface->confirm(GameMessage::kQuitPrompt)) {
_userInterface->notifyShouldExit();
}
2018-07-02 16:32:54 +08:00
} else if (e.kbd.keycode == Common::KEYCODE_p) {
if (_userInterface->isInGameScreen()) {
pauseEngine(true);
}
} else if (e.kbd.keycode == Common::KEYCODE_PAGEUP) {
if (_userInterface->isInventoryOpen()) {
_userInterface->scrollInventoryUp();
}
} else if (e.kbd.keycode == Common::KEYCODE_UP) {
if (_userInterface->isInventoryOpen()) {
_userInterface->scrollInventoryUp();
}
} else if (e.kbd.keycode == Common::KEYCODE_PAGEDOWN) {
if (_userInterface->isInventoryOpen()) {
_userInterface->scrollInventoryDown();
}
} else if (e.kbd.keycode == Common::KEYCODE_DOWN) {
if (_userInterface->isInventoryOpen()) {
_userInterface->scrollInventoryDown();
}
}
} else if (e.type == Common::EVENT_LBUTTONUP) {
2018-05-23 10:25:26 +08:00
_userInterface->handleMouseUp();
} else if (e.type == Common::EVENT_MOUSEMOVE) {
_userInterface->handleMouseMove(e.mouse);
} else if (e.type == Common::EVENT_LBUTTONDOWN) {
_userInterface->handleClick();
if (_system->getMillis() - _lastClickTime < _doubleClickDelay) {
_userInterface->handleDoubleClick();
}
_lastClickTime = _system->getMillis();
} else if (e.type == Common::EVENT_RBUTTONDOWN) {
_userInterface->handleRightClick();
2017-08-26 22:25:18 +02:00
} else if (e.type == Common::EVENT_SCREEN_CHANGED) {
_gfx->computeScreenViewport();
_fontProvider->initFonts();
_userInterface->onScreenChanged();
}
}
}
void StarkEngine::updateDisplayScene() {
if (_global->isFastForward()) {
// The original engine was frame limited to 30 fps.
// Set the frame duration to 1000 / 30 ms so that fast forward
// skips the same amount of simulated time as the original.
_global->setMillisecondsPerGameloop(33);
} else {
_global->setMillisecondsPerGameloop(_frameLimiter->getLastFrameDuration());
}
// Clear the screen
_gfx->clearScreen();
2010-01-21 20:29:34 +10:30
// Only update the world resources when on the game screen
2018-07-02 16:32:54 +08:00
if (_userInterface->isInGameScreen() && !isPaused()) {
int frames = 0;
do {
// Update the game resources
_global->getLevel()->onGameLoop();
_global->getCurrent()->getLevel()->onGameLoop();
_global->getCurrent()->getLocation()->onGameLoop();
frames++;
// When the game is in fast forward mode, update
// the game resources for multiple frames,
// but render only once.
} while (_global->isFastForward() && frames < 100);
_global->setNormalSpeed();
2015-02-26 18:03:12 +01:00
}
// Render the current scene
// Update the UI state before displaying the scene
_userInterface->update();
// Tell the UI to render, and update implicitly, if this leads to new mouse-over events.
_userInterface->render();
2010-01-21 20:29:34 +10:30
}
2014-12-29 23:11:22 +01:00
bool StarkEngine::hasFeature(EngineFeature f) const {
return
(f == kSupportsLoadingDuringRuntime) ||
(f == kSupportsSavingDuringRuntime) ||
2015-12-25 09:40:16 +01:00
(f == kSupportsArbitraryResolutions) ||
(f == kSupportsRTL);
2014-12-29 23:11:22 +01:00
}
bool StarkEngine::canLoadGameStateCurrently() {
return !StarkUserInterface->isInSaveLoadMenuScreen();
2014-12-29 23:11:22 +01:00
}
Common::Error StarkEngine::loadGameState(int slot) {
// Open the save file
Common::String filename = formatSaveName(_targetName.c_str(), slot);
Common::InSaveFile *save = _saveFileMan->openForLoading(filename);
if (!save) {
return _saveFileMan->getError();
}
StateReadStream stream(save);
// Read the header
SaveMetadata metadata;
Common::ErrorCode metadataErrorCode = metadata.read(&stream, filename);
if (metadataErrorCode != Common::kNoError) {
return metadataErrorCode;
}
// Reset the UI
_userInterface->skipFMV();
_userInterface->clearLocationDependentState();
_userInterface->setInteractive(true);
_userInterface->changeScreen(Screen::kScreenGame);
2018-05-21 22:16:13 +08:00
_userInterface->restoreScreenHistory();
// Clear the previous world resources
_resourceProvider->shutdown();
if (metadata.version >= 9) {
metadata.skipGameScreenThumbnail(&stream);
}
// Read the resource trees state
_stateProvider->readStateFromStream(&stream, metadata.version);
2014-12-29 23:11:22 +01:00
// Read the diary state
_diary->readStateFromStream(&stream, metadata.version);
2014-12-29 23:11:22 +01:00
if (stream.eos()) {
warning("Unexpected end of file reached when reading '%s'", filename.c_str());
return Common::kReadingFailed;
}
2014-12-29 23:11:22 +01:00
// Initialize the world resources with the loaded state
2014-12-29 23:11:22 +01:00
_resourceProvider->initGlobal();
_resourceProvider->setShouldRestoreCurrentState();
_resourceProvider->requestLocationChange(metadata.levelIndex, metadata.locationIndex);
if (metadata.version >= 9) {
setTotalPlayTime(metadata.totalPlayTime);
}
2014-12-29 23:11:22 +01:00
return Common::kNoError;
}
bool StarkEngine::canSaveGameStateCurrently() {
// Disallow saving when there is no level loaded or when a script is running
2018-05-31 11:53:51 +08:00
// or when the save & load menu is currently displayed
return _global->getLevel() && _userInterface->isInteractive() && !_userInterface->isInSaveLoadMenuScreen();
2014-12-29 23:11:22 +01:00
}
Common::Error StarkEngine::saveGameState(int slot, const Common::String &desc) {
// Ensure the state store is up to date
_resourceProvider->commitActiveLocationsState();
// Open the save file
Common::String filename = formatSaveName(_targetName.c_str(), slot);
Common::OutSaveFile *save = _saveFileMan->openForSaving(filename);
if (!save) {
return _saveFileMan->getError();
}
// 1. Write the header
SaveMetadata metadata;
metadata.description = desc;
metadata.version = StateProvider::kSaveVersion;
metadata.levelIndex = _global->getCurrent()->getLevel()->getIndex();
metadata.locationIndex = _global->getCurrent()->getLocation()->getIndex();
metadata.totalPlayTime = getTotalPlayTime();
metadata.gameWindowThumbnail = _userInterface->getGameWindowThumbnail();
TimeDate timeDate;
_system->getTimeAndDate(timeDate);
metadata.setSaveTime(timeDate);
2014-12-29 23:11:22 +01:00
metadata.write(save);
metadata.writeGameScreenThumbnail(save);
2014-12-29 23:11:22 +01:00
// 2. Write the resource trees state
2014-12-29 23:11:22 +01:00
_stateProvider->writeStateToStream(save);
// 3. Write the diary state
_diary->writeStateToStream(save);
2014-12-29 23:11:22 +01:00
delete save;
return Common::kNoError;
}
Common::String StarkEngine::formatSaveName(const char *target, int slot) {
return Common::String::format("%s-%03d.tlj", target, slot);
}
void StarkEngine::pauseEngineIntern(bool pause) {
Engine::pauseEngineIntern(pause);
if (!_global || !_global->getLevel() || !_global->getCurrent() || !_frameLimiter) {
// This function may be called when an error occurs before the engine is fully initialized
return;
}
_global->getLevel()->onEnginePause(pause);
_global->getCurrent()->getLevel()->onEnginePause(pause);
_global->getCurrent()->getLocation()->onEnginePause(pause);
_frameLimiter->pause(pause);
// Grab a game screen thumbnail in case we need one when writing a save file
if (_userInterface->isInGameScreen()) {
if (pause) {
_userInterface->saveGameScreenThumbnail();
} else {
_userInterface->freeGameScreenThumbnail();
}
}
// The user may have moved the mouse while the engine was paused
if (!pause) {
StarkUserInterface->handleMouseMove(_eventMan->getMousePos());
}
}
} // End of namespace Stark