2018-10-16 04:47:27 +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.
|
|
|
|
*
|
2018-11-14 04:05:59 +00:00
|
|
|
* This program is free software; you can redistribute it and/or
|
2018-10-16 04:47:27 +00:00
|
|
|
* modify it under the terms of the GNU General Public License
|
2018-11-14 04:05:59 +00:00
|
|
|
* as published by the Free Software Foundation; either version 2
|
2018-10-16 04:47:27 +00:00
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
2018-11-14 04:05:59 +00:00
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
2018-10-16 04:47:27 +00:00
|
|
|
* 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
|
2018-11-14 04:05:59 +00:00
|
|
|
* along with this program; if not, write to the Free Software
|
2018-10-16 04:47:27 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2018-11-14 04:05:59 +00:00
|
|
|
#include "common/scummsys.h"
|
|
|
|
#include "common/config-manager.h"
|
|
|
|
#include "common/debug-channels.h"
|
|
|
|
#include "common/events.h"
|
|
|
|
#include "common/file.h"
|
2019-06-17 01:03:17 +00:00
|
|
|
#include "common/language.h"
|
2018-11-14 04:05:59 +00:00
|
|
|
#include "engines/util.h"
|
|
|
|
#include "graphics/scaler.h"
|
|
|
|
#include "graphics/thumbnail.h"
|
2018-11-14 03:47:07 +00:00
|
|
|
#include "glk/glk.h"
|
2018-11-19 06:32:27 +00:00
|
|
|
#include "glk/blorb.h"
|
2018-11-14 03:47:07 +00:00
|
|
|
#include "glk/conf.h"
|
2020-02-09 05:26:11 +00:00
|
|
|
#include "glk/debugger.h"
|
2018-11-14 03:47:07 +00:00
|
|
|
#include "glk/events.h"
|
|
|
|
#include "glk/picture.h"
|
2018-11-14 04:05:59 +00:00
|
|
|
#include "glk/screen.h"
|
|
|
|
#include "glk/selection.h"
|
2018-12-08 05:52:24 +00:00
|
|
|
#include "glk/sound.h"
|
2020-06-21 23:17:01 +00:00
|
|
|
#include "glk/speech.h"
|
2018-11-14 03:47:07 +00:00
|
|
|
#include "glk/streams.h"
|
|
|
|
#include "glk/windows.h"
|
2018-10-16 04:47:27 +00:00
|
|
|
|
2018-11-14 04:05:59 +00:00
|
|
|
namespace Glk {
|
|
|
|
|
|
|
|
GlkEngine *g_vm;
|
|
|
|
|
2018-11-15 02:01:42 +00:00
|
|
|
GlkEngine::GlkEngine(OSystem *syst, const GlkGameDescription &gameDesc) :
|
2019-11-10 04:55:47 +00:00
|
|
|
_gameDescription(gameDesc), Engine(syst), _random("Glk"), _quitFlag(false), _blorb(nullptr),
|
2020-02-09 05:26:11 +00:00
|
|
|
_clipboard(nullptr), _conf(nullptr),_events(nullptr), _pictures(nullptr), _screen(nullptr),
|
|
|
|
_selection(nullptr), _sounds(nullptr), _streams(nullptr), _windows(nullptr),
|
2019-08-03 02:45:14 +00:00
|
|
|
_copySelect(false), _terminated(false), _pcSpeaker(nullptr), _loadSaveSlot(-1),
|
2019-04-28 04:32:49 +00:00
|
|
|
gli_register_obj(nullptr), gli_unregister_obj(nullptr), gli_register_arr(nullptr),
|
|
|
|
gli_unregister_arr(nullptr) {
|
2019-06-15 22:39:32 +00:00
|
|
|
// Set up debug channels
|
|
|
|
DebugMan.addDebugChannel(kDebugCore, "core", "Core engine debug level");
|
|
|
|
DebugMan.addDebugChannel(kDebugScripts, "scripts", "Game scripts");
|
|
|
|
DebugMan.addDebugChannel(kDebugGraphics, "graphics", "Graphics handling");
|
|
|
|
DebugMan.addDebugChannel(kDebugSound, "sound", "Sound and Music handling");
|
2020-06-21 16:33:55 +00:00
|
|
|
DebugMan.addDebugChannel(kDebugSpeech, "speech", "Text to Speech handling");
|
2019-06-15 22:39:32 +00:00
|
|
|
|
2018-11-14 04:05:59 +00:00
|
|
|
g_vm = this;
|
|
|
|
}
|
|
|
|
|
|
|
|
GlkEngine::~GlkEngine() {
|
2018-11-19 06:32:27 +00:00
|
|
|
delete _blorb;
|
2018-11-14 04:05:59 +00:00
|
|
|
delete _clipboard;
|
|
|
|
delete _events;
|
2019-03-03 04:43:19 +00:00
|
|
|
delete _pcSpeaker;
|
2018-11-19 04:29:14 +00:00
|
|
|
delete _pictures;
|
2018-11-14 04:05:59 +00:00
|
|
|
delete _screen;
|
|
|
|
delete _selection;
|
2018-12-08 05:52:24 +00:00
|
|
|
delete _sounds;
|
2018-11-14 04:05:59 +00:00
|
|
|
delete _streams;
|
|
|
|
delete _windows;
|
2020-08-18 19:06:28 +00:00
|
|
|
delete _conf;
|
2020-06-21 16:15:57 +00:00
|
|
|
|
|
|
|
// Remove all of our debug levels here
|
|
|
|
DebugMan.clearAllDebugChannels();
|
2018-11-14 04:05:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GlkEngine::initialize() {
|
2020-07-05 23:03:48 +00:00
|
|
|
createConfiguration();
|
|
|
|
_conf->load();
|
2020-07-29 04:01:20 +00:00
|
|
|
//_conf->flush();
|
2020-07-05 23:03:48 +00:00
|
|
|
|
2020-07-11 03:43:11 +00:00
|
|
|
initGraphicsMode();
|
|
|
|
createDebugger();
|
|
|
|
|
2018-11-25 20:41:39 +00:00
|
|
|
_screen = createScreen();
|
|
|
|
_screen->initialize();
|
2018-11-14 04:05:59 +00:00
|
|
|
_clipboard = new Clipboard();
|
|
|
|
_events = new Events();
|
2019-03-03 04:43:19 +00:00
|
|
|
_pcSpeaker = new PCSpeaker(_mixer);
|
2018-11-19 04:29:14 +00:00
|
|
|
_pictures = new Pictures();
|
2018-11-14 04:05:59 +00:00
|
|
|
_selection = new Selection();
|
2018-12-08 05:52:24 +00:00
|
|
|
_sounds = new Sounds();
|
2018-11-14 04:05:59 +00:00
|
|
|
_streams = new Streams();
|
|
|
|
_windows = new Windows(_screen);
|
2019-07-06 22:32:39 +00:00
|
|
|
|
|
|
|
// Setup mixer
|
|
|
|
syncSoundSettings();
|
2018-11-14 04:05:59 +00:00
|
|
|
}
|
|
|
|
|
2018-11-25 20:41:39 +00:00
|
|
|
Screen *GlkEngine::createScreen() {
|
|
|
|
return new Screen();
|
|
|
|
}
|
|
|
|
|
2018-11-14 04:05:59 +00:00
|
|
|
void GlkEngine::initGraphicsMode() {
|
2020-07-11 03:43:11 +00:00
|
|
|
initGraphics(_conf->_width, _conf->_height, &_conf->_screenFormat);
|
2018-10-17 02:07:29 +00:00
|
|
|
}
|
|
|
|
|
2020-05-30 14:36:04 +00:00
|
|
|
void GlkEngine::createDebugger() {
|
|
|
|
setDebugger(new Debugger());
|
|
|
|
}
|
|
|
|
|
2020-07-05 23:03:48 +00:00
|
|
|
void GlkEngine::createConfiguration() {
|
|
|
|
_conf = new Conf(getInterpreterType());
|
|
|
|
}
|
|
|
|
|
2018-11-14 04:05:59 +00:00
|
|
|
Common::Error GlkEngine::run() {
|
2019-01-03 07:24:50 +00:00
|
|
|
// Open up the game file
|
2018-11-19 06:32:27 +00:00
|
|
|
Common::String filename = getFilename();
|
|
|
|
if (!Common::File::exists(filename))
|
|
|
|
return Common::kNoGameDataFoundError;
|
|
|
|
|
2018-12-15 03:30:43 +00:00
|
|
|
if (Blorb::isBlorb(filename)) {
|
2018-11-19 06:32:27 +00:00
|
|
|
// Blorb archive
|
|
|
|
_blorb = new Blorb(filename, getInterpreterType());
|
|
|
|
SearchMan.add("blorb", _blorb, 99, false);
|
|
|
|
|
2019-01-03 07:13:13 +00:00
|
|
|
if (!_gameFile.open("game", *_blorb))
|
2018-11-19 06:32:27 +00:00
|
|
|
return Common::kNoGameDataFoundError;
|
|
|
|
} else {
|
2018-12-08 20:18:10 +00:00
|
|
|
// Check for a secondary blorb file with the same filename
|
2019-06-27 04:16:23 +00:00
|
|
|
Common::StringArray blorbFilenames;
|
2019-07-28 02:12:39 +00:00
|
|
|
Blorb::getBlorbFilenames(filename, blorbFilenames, getInterpreterType(), getGameID());
|
2019-06-27 04:16:23 +00:00
|
|
|
|
|
|
|
for (uint idx = 0; idx < blorbFilenames.size(); ++idx) {
|
|
|
|
if (Common::File::exists(blorbFilenames[idx])) {
|
|
|
|
_blorb = new Blorb(blorbFilenames[idx], getInterpreterType());
|
|
|
|
SearchMan.add("blorb", _blorb, 99, false);
|
|
|
|
break;
|
|
|
|
}
|
2018-12-08 20:18:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Open up the game file
|
2019-01-03 07:13:13 +00:00
|
|
|
if (!_gameFile.open(filename))
|
2018-11-19 06:32:27 +00:00
|
|
|
return Common::kNoGameDataFoundError;
|
|
|
|
}
|
|
|
|
|
2019-01-03 07:24:50 +00:00
|
|
|
// Perform initialization
|
|
|
|
initialize();
|
|
|
|
|
|
|
|
// Play the game
|
2020-08-27 21:38:59 +00:00
|
|
|
g_system->setFeatureState(OSystem::kFeatureVirtualKeyboard, true);
|
2019-01-03 07:13:13 +00:00
|
|
|
runGame();
|
2020-08-27 21:38:59 +00:00
|
|
|
g_system->setFeatureState(OSystem::kFeatureVirtualKeyboard, false);
|
2018-10-17 02:07:29 +00:00
|
|
|
|
2018-11-14 04:05:59 +00:00
|
|
|
return Common::kNoError;
|
2018-10-17 02:07:29 +00:00
|
|
|
}
|
|
|
|
|
2020-07-14 02:32:37 +00:00
|
|
|
bool GlkEngine::canLoadGameStateCurrently() {
|
|
|
|
// Only allow savegames by default when sub-engines are waiting for a line
|
|
|
|
Window *win = _windows->getFocusWindow();
|
|
|
|
return win && (win->_lineRequest || win->_lineRequestUni);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GlkEngine::canSaveGameStateCurrently() {
|
|
|
|
// Only allow savegames by default when sub-engines are waiting for a line
|
|
|
|
Window *win = _windows->getFocusWindow();
|
|
|
|
return win && (win->_lineRequest || win->_lineRequestUni);
|
|
|
|
}
|
|
|
|
|
2018-11-18 00:52:47 +00:00
|
|
|
Common::Error GlkEngine::loadGame() {
|
|
|
|
frefid_t ref = _streams->createByPrompt(fileusage_BinaryMode | fileusage_SavedGame,
|
|
|
|
filemode_Read, 0);
|
|
|
|
if (ref == nullptr)
|
|
|
|
return Common::kReadingFailed;
|
|
|
|
|
|
|
|
int slotNumber = ref->_slotNumber;
|
|
|
|
_streams->deleteRef(ref);
|
|
|
|
|
|
|
|
return loadGameState(slotNumber);
|
|
|
|
}
|
|
|
|
|
|
|
|
Common::Error GlkEngine::saveGame() {
|
|
|
|
frefid_t ref = _streams->createByPrompt(fileusage_BinaryMode | fileusage_SavedGame,
|
|
|
|
filemode_Write, 0);
|
|
|
|
if (ref == nullptr)
|
|
|
|
return Common::kWritingFailed;
|
|
|
|
|
|
|
|
int slot = ref->_slotNumber;
|
|
|
|
Common::String desc = ref->_description;
|
|
|
|
_streams->deleteRef(ref);
|
|
|
|
|
|
|
|
return saveGameState(slot, desc);
|
|
|
|
}
|
|
|
|
|
|
|
|
Common::Error GlkEngine::loadGameState(int slot) {
|
|
|
|
FileReference ref(slot, "", fileusage_SavedGame | fileusage_TextMode);
|
|
|
|
|
|
|
|
strid_t file = _streams->openFileStream(&ref, filemode_Read);
|
|
|
|
if (file == nullptr)
|
|
|
|
return Common::kReadingFailed;
|
|
|
|
|
2019-06-16 20:29:15 +00:00
|
|
|
Common::ErrorCode errCode = Common::kNoError;
|
|
|
|
QuetzalReader r;
|
2020-06-25 03:23:11 +00:00
|
|
|
if (r.open(*file, ID_IFSF))
|
|
|
|
// Load in the savegame chunks
|
|
|
|
errCode = loadGameChunks(r).getCode();
|
2018-11-18 00:52:47 +00:00
|
|
|
|
|
|
|
file->close();
|
2019-06-16 20:29:15 +00:00
|
|
|
return errCode;
|
2018-11-18 00:52:47 +00:00
|
|
|
}
|
|
|
|
|
2020-02-05 06:13:33 +00:00
|
|
|
Common::Error GlkEngine::saveGameState(int slot, const Common::String &desc, bool isAutosave) {
|
2018-11-18 00:52:47 +00:00
|
|
|
Common::String msg;
|
|
|
|
FileReference ref(slot, desc, fileusage_BinaryMode | fileusage_SavedGame);
|
|
|
|
|
|
|
|
strid_t file = _streams->openFileStream(&ref, filemode_Write);
|
|
|
|
if (file == nullptr)
|
|
|
|
return Common::kWritingFailed;
|
|
|
|
|
2020-06-25 03:23:11 +00:00
|
|
|
// Write out savegame chunks
|
2019-06-16 20:29:15 +00:00
|
|
|
QuetzalWriter w;
|
2020-06-25 03:23:11 +00:00
|
|
|
Common::ErrorCode errCode = saveGameChunks(w).getCode();
|
2019-06-16 20:29:15 +00:00
|
|
|
|
2019-06-16 20:55:41 +00:00
|
|
|
if (errCode == Common::kNoError) {
|
2019-06-16 20:29:15 +00:00
|
|
|
w.save(*file, desc);
|
|
|
|
}
|
2018-11-18 00:52:47 +00:00
|
|
|
|
|
|
|
file->close();
|
2019-06-16 20:29:15 +00:00
|
|
|
return errCode;
|
2018-11-18 00:52:47 +00:00
|
|
|
}
|
|
|
|
|
2020-06-25 03:23:11 +00:00
|
|
|
Common::Error GlkEngine::loadGameChunks(QuetzalReader &quetzal) {
|
|
|
|
// First scan for a SCVM chunk. It has information of the game the save is for,
|
|
|
|
// so if present we can validate the save is for this game
|
|
|
|
for (QuetzalReader::Iterator it = quetzal.begin(); it != quetzal.end(); ++it) {
|
|
|
|
if ((*it)._id == ID_SCVM) {
|
|
|
|
// Skip over date/time & playtime
|
|
|
|
Common::SeekableReadStream *rs = it.getStream();
|
|
|
|
rs->skip(14);
|
|
|
|
|
|
|
|
uint32 interpType = rs->readUint32BE();
|
|
|
|
Common::String langCode = QuetzalReader::readString(rs);
|
|
|
|
Common::String md5 = QuetzalReader::readString(rs);
|
|
|
|
delete rs;
|
|
|
|
|
|
|
|
if (interpType != QuetzalBase::getInterpreterTag(getInterpreterType()) ||
|
|
|
|
parseLanguage(langCode) != getLanguage() || md5 != getGameMD5())
|
|
|
|
return Common::kReadingFailed;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scan for an uncompressed memory chunk
|
|
|
|
for (QuetzalReader::Iterator it = quetzal.begin(); it != quetzal.end(); ++it) {
|
|
|
|
if ((*it)._id == ID_UMem) {
|
|
|
|
Common::SeekableReadStream *rs = it.getStream();
|
|
|
|
Common::Error err = readSaveData(rs);
|
|
|
|
delete rs;
|
|
|
|
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Couldn't find any data chunk, so reading failed
|
|
|
|
return Common::kReadingFailed;
|
|
|
|
}
|
|
|
|
|
|
|
|
Common::Error GlkEngine::saveGameChunks(QuetzalWriter &quetzal) {
|
|
|
|
// Add the uncompressed memory chunk with the game's save data
|
|
|
|
Common::WriteStream &ws = quetzal.add(ID_UMem);
|
|
|
|
return writeGameData(&ws);
|
|
|
|
}
|
|
|
|
|
2019-07-06 22:32:39 +00:00
|
|
|
void GlkEngine::syncSoundSettings() {
|
|
|
|
Engine::syncSoundSettings();
|
|
|
|
|
|
|
|
int volume = ConfMan.getBool("sfx_mute") ? 0 : CLIP(ConfMan.getInt("sfx_volume"), 0, 255);
|
|
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kPlainSoundType, volume);
|
2020-06-21 23:17:01 +00:00
|
|
|
|
|
|
|
SpeechManager::syncSoundSettings();
|
2019-07-06 22:32:39 +00:00
|
|
|
}
|
|
|
|
|
2019-03-03 04:43:19 +00:00
|
|
|
void GlkEngine::beep() {
|
|
|
|
_pcSpeaker->speakerOn(50, 50);
|
|
|
|
}
|
|
|
|
|
2020-07-08 05:04:29 +00:00
|
|
|
void GlkEngine::switchToWhiteOnBlack() {
|
2020-07-11 03:43:11 +00:00
|
|
|
const uint WHITE = _conf->parseColor("ffffff");
|
|
|
|
const uint BLACK = _conf->parseColor("000000");
|
2020-07-08 05:04:29 +00:00
|
|
|
|
|
|
|
_conf->_wMarginX = 0;
|
|
|
|
_conf->_wMarginY = 0;
|
|
|
|
_conf->_tMarginY = 4;
|
|
|
|
_conf->_propInfo._caretColor = WHITE;
|
|
|
|
|
|
|
|
_conf->_windowColor = _conf->_windowSave = 0;
|
|
|
|
WindowStyle &ws1 = _conf->_tStyles[style_Normal];
|
|
|
|
ws1.bg = BLACK;
|
|
|
|
ws1.fg = WHITE;
|
|
|
|
|
|
|
|
WindowStyle &ws2 = _conf->_tStyles[style_Input];
|
|
|
|
ws2.bg = BLACK;
|
|
|
|
ws2.fg = WHITE;
|
|
|
|
}
|
|
|
|
|
2018-11-14 04:05:59 +00:00
|
|
|
} // End of namespace Glk
|