2014-12-23 15:49:35 +11: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 "common/scummsys.h"
|
|
|
|
#include "common/config-manager.h"
|
|
|
|
#include "common/debug-channels.h"
|
|
|
|
#include "common/events.h"
|
|
|
|
#include "engines/util.h"
|
|
|
|
#include "xeen/xeen.h"
|
2015-01-01 14:57:56 -10:00
|
|
|
#include "xeen/files.h"
|
2015-01-02 16:54:59 -10:00
|
|
|
#include "xeen/resources.h"
|
2014-12-23 15:49:35 +11:00
|
|
|
|
|
|
|
namespace Xeen {
|
|
|
|
|
2016-09-17 20:06:43 -04:00
|
|
|
XeenEngine *g_vm = nullptr;
|
|
|
|
|
2014-12-23 15:49:35 +11:00
|
|
|
XeenEngine::XeenEngine(OSystem *syst, const XeenGameDescription *gameDesc)
|
2015-01-03 03:19:14 -10:00
|
|
|
: Engine(syst), _gameDescription(gameDesc), _randomSource("Xeen") {
|
2016-09-16 19:32:05 -04:00
|
|
|
// Set up debug channels
|
|
|
|
DebugMan.addDebugChannel(kDebugPath, "Path", "Pathfinding debug level");
|
|
|
|
DebugMan.addDebugChannel(kDebugScripts, "scripts", "Game scripts");
|
|
|
|
DebugMan.addDebugChannel(kDebugGraphics, "graphics", "Graphics handling");
|
|
|
|
DebugMan.addDebugChannel(kDebugSound, "sound", "Sound processing");
|
|
|
|
|
2015-01-13 00:16:26 -05:00
|
|
|
_combat = nullptr;
|
2014-12-23 15:49:35 +11:00
|
|
|
_debugger = nullptr;
|
2014-12-25 19:29:38 +11:00
|
|
|
_events = nullptr;
|
2015-01-05 08:11:16 -05:00
|
|
|
_files = nullptr;
|
2015-01-03 03:19:14 -10:00
|
|
|
_interface = nullptr;
|
2017-12-12 20:29:47 -05:00
|
|
|
_locations = nullptr;
|
2015-01-05 08:11:16 -05:00
|
|
|
_map = nullptr;
|
2015-01-19 11:32:57 -05:00
|
|
|
_party = nullptr;
|
2018-04-09 19:33:31 -04:00
|
|
|
_patcher = nullptr;
|
2015-02-05 20:17:16 -05:00
|
|
|
_resources = nullptr;
|
2015-01-02 11:01:41 -10:00
|
|
|
_saves = nullptr;
|
2014-12-25 19:29:38 +11:00
|
|
|
_screen = nullptr;
|
2015-01-19 11:32:57 -05:00
|
|
|
_scripts = nullptr;
|
2014-12-26 14:37:20 +11:00
|
|
|
_sound = nullptr;
|
2015-01-25 21:19:59 -05:00
|
|
|
_spells = nullptr;
|
2017-11-29 20:24:03 -05:00
|
|
|
_windows = nullptr;
|
2015-01-02 16:54:59 -10:00
|
|
|
_noDirectionSense = false;
|
2017-11-15 19:32:03 -05:00
|
|
|
_startupWindowActive = false;
|
2018-02-26 19:15:00 -05:00
|
|
|
_gameMode = GMODE_STARTUP;
|
2018-03-13 18:17:02 -04:00
|
|
|
_mode = MODE_STARTUP;
|
2016-09-19 23:01:32 -04:00
|
|
|
_endingScore = 0;
|
2017-11-15 19:32:03 -05:00
|
|
|
_loadSaveSlot = -1;
|
2018-03-04 22:21:51 -05:00
|
|
|
_gameWon[0] = _gameWon[1] = _gameWon[2] = false;
|
|
|
|
_finalScore = 0;
|
2016-09-17 20:06:43 -04:00
|
|
|
g_vm = this;
|
2014-12-23 15:49:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
XeenEngine::~XeenEngine() {
|
2015-01-13 00:16:26 -05:00
|
|
|
delete _combat;
|
2014-12-23 15:49:35 +11:00
|
|
|
delete _debugger;
|
2014-12-25 19:29:38 +11:00
|
|
|
delete _events;
|
2015-01-03 03:19:14 -10:00
|
|
|
delete _interface;
|
2017-12-12 20:29:47 -05:00
|
|
|
delete _locations;
|
2015-01-05 08:11:16 -05:00
|
|
|
delete _map;
|
2015-01-19 11:32:57 -05:00
|
|
|
delete _party;
|
2018-04-09 19:33:31 -04:00
|
|
|
delete _patcher;
|
2015-01-02 11:01:41 -10:00
|
|
|
delete _saves;
|
2014-12-25 19:29:38 +11:00
|
|
|
delete _screen;
|
2015-01-19 11:32:57 -05:00
|
|
|
delete _scripts;
|
2014-12-26 14:37:20 +11:00
|
|
|
delete _sound;
|
2015-01-25 21:19:59 -05:00
|
|
|
delete _spells;
|
2017-11-29 20:24:03 -05:00
|
|
|
delete _windows;
|
2015-02-05 20:17:16 -05:00
|
|
|
delete _resources;
|
2015-01-05 08:11:16 -05:00
|
|
|
delete _files;
|
2016-09-17 20:06:43 -04:00
|
|
|
g_vm = nullptr;
|
2014-12-23 15:49:35 +11:00
|
|
|
}
|
|
|
|
|
2018-02-20 18:50:17 -05:00
|
|
|
bool XeenEngine::initialize() {
|
2014-12-23 15:49:35 +11:00
|
|
|
// Create sub-objects of the engine
|
2015-01-05 08:11:16 -05:00
|
|
|
_files = new FileManager(this);
|
2018-02-20 18:50:17 -05:00
|
|
|
if (!_files->setup())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
_resources = new Resources();
|
2015-01-13 00:16:26 -05:00
|
|
|
_combat = new Combat(this);
|
2014-12-23 15:49:35 +11:00
|
|
|
_debugger = new Debugger(this);
|
2014-12-25 19:29:38 +11:00
|
|
|
_events = new EventsManager(this);
|
2015-01-03 03:19:14 -10:00
|
|
|
_interface = new Interface(this);
|
2017-12-12 20:29:47 -05:00
|
|
|
_locations = new LocationManager();
|
2015-01-05 08:11:16 -05:00
|
|
|
_map = new Map(this);
|
2015-01-19 11:32:57 -05:00
|
|
|
_party = new Party(this);
|
2018-04-09 19:33:31 -04:00
|
|
|
_patcher = new Patcher();
|
2017-12-22 08:52:31 -05:00
|
|
|
_saves = new SavesManager(_targetName);
|
2014-12-25 19:29:38 +11:00
|
|
|
_screen = new Screen(this);
|
2015-01-19 11:32:57 -05:00
|
|
|
_scripts = new Scripts(this);
|
2018-02-19 10:17:33 -05:00
|
|
|
_sound = new Sound(_mixer);
|
2015-01-25 21:19:59 -05:00
|
|
|
_spells = new Spells(this);
|
2017-11-29 20:24:03 -05:00
|
|
|
_windows = new Windows();
|
2014-12-23 15:49:35 +11:00
|
|
|
|
|
|
|
// Set graphics mode
|
2018-02-26 19:20:34 -05:00
|
|
|
initGraphics(SCREEN_WIDTH, SCREEN_HEIGHT);
|
2014-12-23 15:49:35 +11:00
|
|
|
|
2018-02-18 21:48:50 -05:00
|
|
|
// Setup mixer
|
|
|
|
syncSoundSettings();
|
|
|
|
|
2018-03-03 22:15:50 -05:00
|
|
|
// Load settings
|
2018-03-25 23:12:14 -04:00
|
|
|
loadSettings();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void XeenEngine::loadSettings() {
|
2018-03-03 22:15:50 -05:00
|
|
|
_gameWon[0] = ConfMan.hasKey("game_won") && ConfMan.getBool("game_won");
|
|
|
|
_gameWon[1] = ConfMan.hasKey("game_won2") && ConfMan.getBool("game_won2");
|
2018-03-04 22:21:51 -05:00
|
|
|
_gameWon[2] = ConfMan.hasKey("game_won3") && ConfMan.getBool("game_won3");
|
|
|
|
_finalScore = ConfMan.hasKey("final_score") ? ConfMan.getInt("final_score") : 0;
|
2018-03-03 22:15:50 -05:00
|
|
|
|
2018-03-25 23:12:14 -04:00
|
|
|
_extOptions._showItemCosts = ConfMan.hasKey("ShowItemCosts") && ConfMan.getBool("ShowItemCosts");
|
2018-04-13 22:56:37 -04:00
|
|
|
_extOptions._durableArmor = ConfMan.hasKey("DurableArmor") && ConfMan.getBool("DurableArmor");
|
2018-03-25 23:12:14 -04:00
|
|
|
|
2014-12-23 15:49:35 +11:00
|
|
|
// If requested, load a savegame instead of showing the intro
|
|
|
|
if (ConfMan.hasKey("save_slot")) {
|
|
|
|
int saveSlot = ConfMan.getInt("save_slot");
|
|
|
|
if (saveSlot >= 0 && saveSlot <= 999)
|
|
|
|
_loadSaveSlot = saveSlot;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Common::Error XeenEngine::run() {
|
2018-02-20 18:50:17 -05:00
|
|
|
if (initialize())
|
|
|
|
outerGameLoop();
|
2014-12-23 15:49:35 +11:00
|
|
|
|
|
|
|
return Common::kNoError;
|
|
|
|
}
|
|
|
|
|
2018-02-26 19:15:00 -05:00
|
|
|
void XeenEngine::outerGameLoop() {
|
|
|
|
if (_loadSaveSlot != -1)
|
|
|
|
// Loading savegame from launcher, so Skip menu and go straight to game
|
|
|
|
_gameMode = GMODE_PLAY_GAME;
|
|
|
|
|
|
|
|
while (!shouldQuit() && _gameMode != GMODE_QUIT) {
|
|
|
|
GameMode mode = _gameMode;
|
|
|
|
_gameMode = GMODE_NONE;
|
|
|
|
assert(mode != GMODE_NONE);
|
|
|
|
|
|
|
|
switch (mode) {
|
|
|
|
case GMODE_STARTUP:
|
|
|
|
showStartup();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GMODE_MENU:
|
|
|
|
showMainMenu();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GMODE_PLAY_GAME:
|
|
|
|
playGame();
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-23 15:49:35 +11:00
|
|
|
int XeenEngine::getRandomNumber(int maxNumber) {
|
|
|
|
return _randomSource.getRandomNumber(maxNumber);
|
|
|
|
}
|
|
|
|
|
2015-01-22 22:05:36 -05:00
|
|
|
int XeenEngine::getRandomNumber(int minNumber, int maxNumber) {
|
|
|
|
return getRandomNumber(maxNumber - minNumber) + minNumber;
|
|
|
|
}
|
|
|
|
|
2014-12-23 15:49:35 +11:00
|
|
|
Common::Error XeenEngine::saveGameState(int slot, const Common::String &desc) {
|
2017-12-22 08:52:31 -05:00
|
|
|
return _saves->saveGameState(slot, desc);
|
2014-12-23 15:49:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
Common::Error XeenEngine::loadGameState(int slot) {
|
2018-01-14 17:59:43 -05:00
|
|
|
_loadSaveSlot = slot;
|
|
|
|
return Common::kNoError;
|
2014-12-23 15:49:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
bool XeenEngine::canLoadGameStateCurrently() {
|
2018-04-07 15:53:27 -04:00
|
|
|
return _mode != MODE_STARTUP;
|
2014-12-23 15:49:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
bool XeenEngine::canSaveGameStateCurrently() {
|
2018-04-15 09:51:18 -04:00
|
|
|
return _mode != MODE_COMBAT && _mode != MODE_STARTUP && _mode != MODE_SCRIPT_IN_PROGRESS
|
|
|
|
&& (_map->mazeData()._mazeFlags & RESTRICTION_SAVE) == 0;
|
2014-12-23 15:49:35 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
void XeenEngine::playGame() {
|
2017-12-20 21:47:16 -05:00
|
|
|
_files->setGameCc(0);
|
2016-09-23 13:11:29 -04:00
|
|
|
_sound->stopAllAudio();
|
2018-03-09 21:47:28 -05:00
|
|
|
SpriteResource::setClippedBottom(140);
|
2016-09-23 13:11:29 -04:00
|
|
|
|
2015-01-05 08:11:16 -05:00
|
|
|
play();
|
2018-03-31 22:36:57 -04:00
|
|
|
_sound->stopAllAudio();
|
2015-01-05 08:11:16 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void XeenEngine::play() {
|
2015-01-05 21:17:44 -05:00
|
|
|
_interface->setup();
|
2015-01-05 08:11:16 -05:00
|
|
|
_screen->loadBackground("back.raw");
|
|
|
|
_screen->loadPalette("mm4.pal");
|
|
|
|
|
2018-03-27 21:45:17 -04:00
|
|
|
if (getGameID() == GType_DarkSide && !_map->_loadCcNum) {
|
|
|
|
_map->_loadCcNum = 1;
|
2015-01-19 11:32:57 -05:00
|
|
|
_party->_mazeId = 29;
|
|
|
|
_party->_mazeDirection = DIR_NORTH;
|
|
|
|
_party->_mazePosition.x = 25;
|
|
|
|
_party->_mazePosition.y = 21;
|
2015-01-05 08:11:16 -05:00
|
|
|
}
|
|
|
|
|
2018-04-02 21:36:36 -04:00
|
|
|
_map->clearMaze();
|
2018-01-14 17:59:43 -05:00
|
|
|
if (_loadSaveSlot >= 0) {
|
2018-03-03 22:15:50 -05:00
|
|
|
_saves->newGame();
|
2018-01-14 17:59:43 -05:00
|
|
|
_saves->loadGameState(_loadSaveSlot);
|
|
|
|
_loadSaveSlot = -1;
|
|
|
|
} else {
|
|
|
|
_map->load(_party->_mazeId);
|
|
|
|
}
|
2015-01-05 15:10:42 -05:00
|
|
|
|
|
|
|
_interface->startup();
|
2018-03-13 18:17:02 -04:00
|
|
|
if (_mode == MODE_STARTUP) {
|
2016-09-19 00:01:25 -04:00
|
|
|
// _screen->fadeOut();
|
2015-01-05 15:10:42 -05:00
|
|
|
}
|
|
|
|
|
2017-11-29 20:24:03 -05:00
|
|
|
(*_windows)[0].update();
|
2015-01-05 15:10:42 -05:00
|
|
|
_interface->mainIconsPrint();
|
2017-11-29 20:24:03 -05:00
|
|
|
(*_windows)[0].update();
|
2015-01-05 15:10:42 -05:00
|
|
|
_events->setCursor(0);
|
|
|
|
|
2015-02-22 23:45:11 -05:00
|
|
|
_combat->_moveMonsters = true;
|
2018-03-13 18:17:02 -04:00
|
|
|
if (_mode == MODE_STARTUP) {
|
2018-04-15 09:51:18 -04:00
|
|
|
_mode = MODE_INTERACTIVE;
|
2016-09-19 00:01:25 -04:00
|
|
|
_screen->fadeIn();
|
2015-01-05 15:10:42 -05:00
|
|
|
}
|
|
|
|
|
2015-02-22 23:45:11 -05:00
|
|
|
_combat->_moveMonsters = true;
|
2015-01-06 19:09:07 -05:00
|
|
|
|
2015-01-19 11:32:57 -05:00
|
|
|
gameLoop();
|
2018-01-18 21:29:42 -05:00
|
|
|
|
|
|
|
if (_party->_dead)
|
|
|
|
death();
|
2018-03-13 18:17:02 -04:00
|
|
|
|
|
|
|
_mode = MODE_STARTUP;
|
2018-03-16 21:41:19 -04:00
|
|
|
_gameMode = GMODE_MENU;
|
2015-01-19 11:32:57 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void XeenEngine::gameLoop() {
|
2015-01-05 15:10:42 -05:00
|
|
|
// Main game loop
|
2018-04-07 15:53:27 -04:00
|
|
|
while (isLoadPending() || !shouldExit()) {
|
2018-03-20 22:06:22 -04:00
|
|
|
if (isLoadPending()) {
|
2018-01-14 17:59:43 -05:00
|
|
|
// Load any pending savegame
|
|
|
|
int saveSlot = _loadSaveSlot;
|
|
|
|
_loadSaveSlot = -1;
|
2019-08-18 15:18:48 -07:00
|
|
|
(void)_saves->loadGameState(saveSlot);
|
2018-03-23 21:43:16 -04:00
|
|
|
_interface->drawParty(true);
|
2018-01-14 17:59:43 -05:00
|
|
|
}
|
|
|
|
|
2015-01-19 11:32:57 -05:00
|
|
|
_map->cellFlagLookup(_party->_mazePosition);
|
|
|
|
if (_map->_currentIsEvent) {
|
2018-02-26 19:15:00 -05:00
|
|
|
_gameMode = (GameMode)_scripts->checkEvents();
|
2018-04-15 09:51:18 -04:00
|
|
|
if (isLoadPending())
|
|
|
|
continue;
|
2018-04-07 15:53:27 -04:00
|
|
|
if (shouldExit())
|
2015-01-19 11:32:57 -05:00
|
|
|
return;
|
|
|
|
}
|
2015-02-27 07:50:32 -05:00
|
|
|
_party->giveTreasure();
|
2015-01-19 11:32:57 -05:00
|
|
|
|
2015-01-19 12:13:03 -05:00
|
|
|
// Main user interface handler for waiting for and processing user input
|
|
|
|
_interface->perform();
|
2018-01-20 17:32:10 -05:00
|
|
|
|
|
|
|
if (_party->_dead)
|
|
|
|
break;
|
2015-01-05 15:10:42 -05:00
|
|
|
}
|
2015-01-02 16:02:24 -10:00
|
|
|
}
|
|
|
|
|
2015-01-26 21:35:50 -05:00
|
|
|
Common::String XeenEngine::printMil(uint value) {
|
2016-09-04 14:19:05 -04:00
|
|
|
return (value >= 1000000) ? Common::String::format("%u mil", value / 1000000) :
|
|
|
|
Common::String::format("%u", value);
|
2015-01-26 21:35:50 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
Common::String XeenEngine::printK(uint value) {
|
|
|
|
return (value > 9999) ? Common::String::format("%uk", value / 1000) :
|
|
|
|
Common::String::format("%u", value);
|
|
|
|
}
|
|
|
|
|
2015-03-01 10:41:11 -05:00
|
|
|
Common::String XeenEngine::printK2(uint value) {
|
|
|
|
return (value > 999) ? Common::String::format("%uk", value / 1000) :
|
|
|
|
Common::String::format("%u", value);
|
|
|
|
}
|
|
|
|
|
2018-01-28 11:53:11 -05:00
|
|
|
void XeenEngine::syncSoundSettings() {
|
2018-02-18 21:48:50 -05:00
|
|
|
Engine::syncSoundSettings();
|
|
|
|
|
2018-01-28 11:53:11 -05:00
|
|
|
if (_sound)
|
|
|
|
_sound->updateSoundSettings();
|
|
|
|
}
|
|
|
|
|
2018-03-03 22:15:50 -05:00
|
|
|
void XeenEngine::saveSettings() {
|
|
|
|
if (_gameWon[0])
|
|
|
|
ConfMan.setBool("game_won", true);
|
|
|
|
if (_gameWon[1])
|
|
|
|
ConfMan.setBool("game_won2", true);
|
2018-03-04 22:21:51 -05:00
|
|
|
if (_gameWon[2])
|
|
|
|
ConfMan.setBool("game_won3", true);
|
2018-03-03 22:15:50 -05:00
|
|
|
|
2018-03-04 22:21:51 -05:00
|
|
|
ConfMan.setInt("final_score", _finalScore);
|
2018-03-03 22:15:50 -05:00
|
|
|
ConfMan.flushToDisk();
|
|
|
|
}
|
|
|
|
|
2019-08-18 15:18:48 -07:00
|
|
|
void XeenEngine::GUIError(const Common::String &msg) {
|
|
|
|
GUIErrorMessage(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
void XeenEngine::autoSaveCheck(int &lastSaveTime) {
|
|
|
|
if (shouldPerformAutoSave(lastSaveTime) && canSaveGameStateCurrently() &&
|
|
|
|
(_map && !(_map->mazeData()._mazeFlags & RESTRICTION_SAVE))) {
|
|
|
|
_saves->doAutosave();
|
|
|
|
lastSaveTime = g_system->getMillis();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-23 15:49:35 +11:00
|
|
|
} // End of namespace Xeen
|