scummvm/engines/glk/glk.cpp
2019-11-11 18:20:30 -08:00

268 lines
7.7 KiB
C++

/* 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 "common/file.h"
#include "common/language.h"
#include "engines/util.h"
#include "graphics/scaler.h"
#include "graphics/thumbnail.h"
#include "glk/glk.h"
#include "glk/blorb.h"
#include "glk/conf.h"
#include "glk/events.h"
#include "glk/picture.h"
#include "glk/quetzal.h"
#include "glk/screen.h"
#include "glk/selection.h"
#include "glk/sound.h"
#include "glk/streams.h"
#include "glk/windows.h"
namespace Glk {
GlkEngine *g_vm;
GlkEngine::GlkEngine(OSystem *syst, const GlkGameDescription &gameDesc) :
_gameDescription(gameDesc), Engine(syst), _random("Glk"), _quitFlag(false), _blorb(nullptr),
_clipboard(nullptr), _conf(nullptr), _debugger(nullptr), _events(nullptr), _pictures(nullptr),
_screen(nullptr), _selection(nullptr), _sounds(nullptr), _streams(nullptr), _windows(nullptr),
_copySelect(false), _terminated(false), _pcSpeaker(nullptr), _loadSaveSlot(-1),
gli_register_obj(nullptr), gli_unregister_obj(nullptr), gli_register_arr(nullptr),
gli_unregister_arr(nullptr) {
// 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");
g_vm = this;
}
GlkEngine::~GlkEngine() {
delete _blorb;
delete _clipboard;
delete _conf;
delete _debugger;
delete _events;
delete _pcSpeaker;
delete _pictures;
delete _screen;
delete _selection;
delete _sounds;
delete _streams;
delete _windows;
}
void GlkEngine::initialize() {
initGraphicsMode();
_conf = new Conf(getInterpreterType());
_debugger = createDebugger();
_screen = createScreen();
_screen->initialize();
_clipboard = new Clipboard();
_events = new Events();
_pcSpeaker = new PCSpeaker(_mixer);
_pictures = new Pictures();
_selection = new Selection();
_sounds = new Sounds();
_streams = new Streams();
_windows = new Windows(_screen);
// Setup mixer
syncSoundSettings();
}
Screen *GlkEngine::createScreen() {
return new Screen();
}
void GlkEngine::initGraphicsMode() {
uint width = ConfMan.hasKey("width") ? ConfMan.getInt("width") : 640;
uint height = ConfMan.hasKey("height") ? ConfMan.getInt("height") : 480;
Common::List<Graphics::PixelFormat> formats = g_system->getSupportedFormats();
Graphics::PixelFormat format = formats.front();
for (Common::List<Graphics::PixelFormat>::iterator i = formats.begin(); i != formats.end(); ++i) {
if ((*i).bytesPerPixel > 1) {
format = *i;
break;
}
}
initGraphics(width, height, &format);
}
Common::Error GlkEngine::run() {
// Open up the game file
Common::String filename = getFilename();
if (!Common::File::exists(filename))
return Common::kNoGameDataFoundError;
if (Blorb::isBlorb(filename)) {
// Blorb archive
_blorb = new Blorb(filename, getInterpreterType());
SearchMan.add("blorb", _blorb, 99, false);
if (!_gameFile.open("game", *_blorb))
return Common::kNoGameDataFoundError;
} else {
// Check for a secondary blorb file with the same filename
Common::StringArray blorbFilenames;
Blorb::getBlorbFilenames(filename, blorbFilenames, getInterpreterType(), getGameID());
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;
}
}
// Open up the game file
if (!_gameFile.open(filename))
return Common::kNoGameDataFoundError;
}
// Perform initialization
initialize();
// Play the game
runGame();
return Common::kNoError;
}
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;
Common::ErrorCode errCode = Common::kNoError;
QuetzalReader r;
if (r.open(*file, ID_IFSF)) {
// 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 = r.begin(); it != r.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())
errCode = Common::kReadingFailed;
}
}
if (errCode == Common::kNoError) {
// Scan for an uncompressed memory chunk
errCode = Common::kReadingFailed; // Presume we won't find chunk
for (QuetzalReader::Iterator it = r.begin(); it != r.end(); ++it) {
if ((*it)._id == ID_UMem) {
Common::SeekableReadStream *rs = it.getStream();
errCode = readSaveData(rs).getCode();
delete rs;
break;
}
}
}
}
file->close();
return errCode;
}
Common::Error GlkEngine::saveGameState(int slot, const Common::String &desc) {
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;
Common::ErrorCode errCode = Common::kNoError;
QuetzalWriter w;
// Add the uncompressed memory chunk with the game's save data
{
Common::WriteStream &ws = w.add(ID_UMem);
errCode = writeGameData(&ws).getCode();
}
if (errCode == Common::kNoError) {
w.save(*file, desc);
}
file->close();
return errCode;
}
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);
}
void GlkEngine::beep() {
_pcSpeaker->speakerOn(50, 50);
}
} // End of namespace Glk