2020-12-24 16:10:32 -03:00
|
|
|
#include "common/scummsys.h"
|
|
|
|
|
2020-12-24 19:02:12 -03:00
|
|
|
#include "audio/decoders/wave.h"
|
|
|
|
#include "audio/audiostream.h"
|
|
|
|
|
2021-01-13 19:27:09 -03:00
|
|
|
#include "common/archive.h"
|
2020-12-24 16:10:32 -03:00
|
|
|
#include "common/config-manager.h"
|
|
|
|
#include "common/debug.h"
|
|
|
|
#include "common/debug-channels.h"
|
|
|
|
#include "common/error.h"
|
|
|
|
#include "common/events.h"
|
|
|
|
#include "common/file.h"
|
|
|
|
#include "common/fs.h"
|
|
|
|
#include "common/system.h"
|
2021-01-02 14:19:23 -03:00
|
|
|
#include "common/str.h"
|
2021-01-12 20:49:12 -03:00
|
|
|
#include "common/savefile.h"
|
2020-12-24 16:10:32 -03:00
|
|
|
#include "engines/util.h"
|
|
|
|
|
2021-01-02 14:19:23 -03:00
|
|
|
#include "image/bmp.h"
|
2021-01-07 23:10:19 -03:00
|
|
|
#include "graphics/cursorman.h"
|
2021-01-02 14:19:23 -03:00
|
|
|
|
2021-01-10 11:18:55 -03:00
|
|
|
#include "private/cursors.h"
|
2020-12-24 16:10:32 -03:00
|
|
|
#include "private/private.h"
|
2020-12-30 14:34:32 -03:00
|
|
|
#include "private/grammar.tab.h"
|
|
|
|
#include "private/grammar.h"
|
|
|
|
|
2021-01-10 11:18:55 -03:00
|
|
|
|
2020-12-24 16:10:32 -03:00
|
|
|
namespace Private {
|
|
|
|
|
2021-01-07 23:32:17 -03:00
|
|
|
PrivateEngine *g_private = NULL;
|
2021-01-07 23:10:19 -03:00
|
|
|
|
2020-12-30 17:25:02 -03:00
|
|
|
extern int parse(char*);
|
|
|
|
|
2020-12-24 16:10:32 -03:00
|
|
|
PrivateEngine::PrivateEngine(OSystem *syst)
|
2021-01-07 23:38:18 -03:00
|
|
|
: Engine(syst) {
|
|
|
|
// Put your engine in a sane state, but do nothing big yet;
|
|
|
|
// in particular, do not load data from files; rather, if you
|
|
|
|
// need to do such things, do them from run().
|
2020-12-24 16:10:32 -03:00
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
// Do not initialize graphics here
|
|
|
|
// Do not initialize audio devices here
|
2020-12-24 16:10:32 -03:00
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
// However this is the place to specify all default directories
|
2021-01-13 19:27:09 -03:00
|
|
|
//SearchMan.addSubDirectoryMatching(gameDataDir, "global", 0, 10, false);
|
2020-12-24 16:10:32 -03:00
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
// Here is the right place to set up the engine specific debug channels
|
|
|
|
//DebugMan.addDebugChannel(kPrivateDebugExample, "example", "this is just an example for a engine specific debug channel");
|
|
|
|
//DebugMan.addDebugChannel(kPrivateDebugExample2, "example2", "also an example");
|
2020-12-24 16:10:32 -03:00
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
// Don't forget to register your random source
|
|
|
|
_rnd = new Common::RandomSource("private");
|
2020-12-24 16:10:32 -03:00
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
g_private = this;
|
2021-01-07 23:32:17 -03:00
|
|
|
|
2021-01-12 20:49:12 -03:00
|
|
|
_saveGameMask = NULL;
|
|
|
|
_loadGameMask = NULL;
|
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
_nextSetting = NULL;
|
|
|
|
_nextMovie = NULL;
|
2021-01-16 15:30:00 -03:00
|
|
|
_nextVS = NULL;
|
2021-01-07 23:38:18 -03:00
|
|
|
_modified = false;
|
|
|
|
_mode = -1;
|
2021-01-09 16:11:30 -03:00
|
|
|
_frame = new Common::String("inface/general/inface2.bmp");
|
|
|
|
|
2021-01-16 15:30:00 -03:00
|
|
|
_policeRadioArea = NULL;
|
|
|
|
_AMRadioArea = NULL;
|
|
|
|
_phoneArea = NULL;
|
|
|
|
|
|
|
|
_AMRadioPrefix = new Common::String("inface/radio/comm_/");
|
|
|
|
_policeRadioPrefix = new Common::String("inface/radio/police/");
|
|
|
|
_phonePrefix = new Common::String("inface/telephon/");
|
|
|
|
_phoneCallSound = new Common::String("phone.wav");
|
2021-01-07 23:32:17 -03:00
|
|
|
|
2020-12-24 16:10:32 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
PrivateEngine::~PrivateEngine() {
|
2021-01-07 23:38:18 -03:00
|
|
|
debug("PrivateEngine::~PrivateEngine");
|
2020-12-24 16:10:32 -03:00
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
// Dispose your resources here
|
|
|
|
delete _rnd;
|
2020-12-24 16:10:32 -03:00
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
// Remove all of our debug levels here
|
|
|
|
DebugMan.clearAllDebugChannels();
|
2020-12-24 16:10:32 -03:00
|
|
|
}
|
|
|
|
|
2021-01-13 19:27:09 -03:00
|
|
|
void PrivateEngine::initializePath(const Common::FSNode &gamePath) {
|
|
|
|
SearchMan.addDirectory(gamePath.getPath(), gamePath, 0, 10);
|
|
|
|
}
|
|
|
|
|
2020-12-24 16:10:32 -03:00
|
|
|
Common::Error PrivateEngine::run() {
|
2021-01-13 19:27:09 -03:00
|
|
|
|
2021-01-10 20:17:27 -03:00
|
|
|
assert(_installerArchive.open("SUPPORT/ASSETS.Z"));
|
|
|
|
Common::SeekableReadStream *file = NULL;
|
2021-01-10 19:11:02 -03:00
|
|
|
|
2021-01-16 15:30:00 -03:00
|
|
|
// if the full game is used
|
|
|
|
if (_installerArchive.hasFile("GAME.DAT"))
|
2021-01-10 20:17:27 -03:00
|
|
|
file = _installerArchive.createReadStreamForMember("GAME.DAT");
|
2021-01-16 15:30:00 -03:00
|
|
|
|
|
|
|
// if the demo from archive.org is used
|
|
|
|
else if (_installerArchive.hasFile("GAME.TXT"))
|
2021-01-10 20:17:27 -03:00
|
|
|
file = _installerArchive.createReadStreamForMember("GAME.TXT");
|
2021-01-16 15:30:00 -03:00
|
|
|
|
|
|
|
// if the demo from the full retail CDROM is used
|
|
|
|
else if (_installerArchive.hasFile("DEMOGAME.DAT"))
|
2021-01-14 18:48:30 -03:00
|
|
|
file = _installerArchive.createReadStreamForMember("DEMOGAME.DAT");
|
2021-01-09 19:11:41 -03:00
|
|
|
|
2021-01-10 19:11:02 -03:00
|
|
|
assert(file != NULL);
|
2021-01-07 23:38:18 -03:00
|
|
|
void *buf = malloc(191000);
|
|
|
|
file->read(buf, 191000);
|
2021-01-10 11:18:55 -03:00
|
|
|
|
|
|
|
// Initialize stuff
|
2021-01-09 22:08:41 -03:00
|
|
|
initInsts();
|
2021-01-07 23:38:18 -03:00
|
|
|
initFuncs();
|
2021-01-10 11:18:55 -03:00
|
|
|
initCursors();
|
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
parse((char *) buf);
|
|
|
|
assert(constants.size() > 0);
|
|
|
|
|
|
|
|
// Initialize graphics using following:
|
|
|
|
_screenW = 640;
|
|
|
|
_screenH = 480;
|
|
|
|
//_pixelFormat = Graphics::PixelFormat::createFormatCLUT8();
|
|
|
|
_pixelFormat = Graphics::PixelFormat(4, 8, 8, 8, 8, 24, 16, 8, 0);
|
2021-01-08 22:42:34 -03:00
|
|
|
_transparentColor = _pixelFormat.RGBToColor(0,255,0);
|
2021-01-07 23:38:18 -03:00
|
|
|
initGraphics(_screenW, _screenH, &_pixelFormat);
|
2021-01-14 18:48:30 -03:00
|
|
|
changeCursor("default");
|
2021-01-10 19:11:02 -03:00
|
|
|
|
2021-01-08 22:42:34 -03:00
|
|
|
_origin = new Common::Point(0, 0);
|
2021-01-07 23:38:18 -03:00
|
|
|
_image = new Image::BitmapDecoder();
|
|
|
|
_compositeSurface = new Graphics::ManagedSurface();
|
|
|
|
_compositeSurface->create(_screenW, _screenH, _pixelFormat);
|
2021-01-08 22:42:34 -03:00
|
|
|
_compositeSurface->setTransparentColor(_transparentColor);
|
2021-01-07 23:38:18 -03:00
|
|
|
|
|
|
|
// Create debugger console. It requires GFX to be initialized
|
|
|
|
Console *console = new Console(this);
|
|
|
|
setDebugger(console);
|
|
|
|
|
|
|
|
// Additional setup.
|
|
|
|
debug("PrivateEngine::init");
|
|
|
|
|
|
|
|
// Simple main event loop
|
|
|
|
Common::Event event;
|
|
|
|
Common::Point mousePos;
|
|
|
|
_videoDecoder = nullptr; //new Video::SmackerDecoder();
|
|
|
|
|
2021-01-13 19:27:09 -03:00
|
|
|
int saveSlot = ConfMan.getInt("save_slot");
|
2021-01-12 20:49:12 -03:00
|
|
|
if (saveSlot >= 0) { // load the savegame
|
|
|
|
loadGameState(saveSlot);
|
|
|
|
} else {
|
|
|
|
_nextSetting = new Common::String("kGoIntro");
|
|
|
|
}
|
2021-01-07 23:38:18 -03:00
|
|
|
|
|
|
|
while (!shouldQuit()) {
|
2021-01-09 17:00:11 -03:00
|
|
|
while (g_system->getEventManager()->pollEvent(event)) {
|
2021-01-10 11:18:55 -03:00
|
|
|
mousePos = g_system->getEventManager()->getMousePos();
|
2021-01-09 17:00:11 -03:00
|
|
|
// Events
|
|
|
|
switch (event.type) {
|
2021-01-09 19:11:41 -03:00
|
|
|
case Common::EVENT_KEYDOWN:
|
|
|
|
if (event.kbd.keycode == Common::KEYCODE_ESCAPE && _videoDecoder)
|
|
|
|
skipVideo();
|
|
|
|
break;
|
2021-01-09 17:00:11 -03:00
|
|
|
|
2021-01-09 19:11:41 -03:00
|
|
|
case Common::EVENT_QUIT:
|
|
|
|
case Common::EVENT_RETURN_TO_LAUNCHER:
|
|
|
|
break;
|
2021-01-09 17:00:11 -03:00
|
|
|
|
2021-01-09 19:11:41 -03:00
|
|
|
case Common::EVENT_LBUTTONDOWN:
|
2021-01-16 15:30:00 -03:00
|
|
|
selectPoliceRadioArea(mousePos);
|
|
|
|
selectAMRadioArea(mousePos);
|
2021-01-12 20:49:12 -03:00
|
|
|
selectLoadGame(mousePos);
|
|
|
|
selectSaveGame(mousePos);
|
2021-01-09 19:11:41 -03:00
|
|
|
selectMask(mousePos);
|
|
|
|
if (!_nextSetting)
|
|
|
|
selectExit(mousePos);
|
|
|
|
break;
|
2021-01-09 17:00:11 -03:00
|
|
|
|
2021-01-10 11:18:55 -03:00
|
|
|
case Common::EVENT_MOUSEMOVE:
|
2021-01-14 18:48:30 -03:00
|
|
|
changeCursor("default");
|
2021-01-16 15:30:00 -03:00
|
|
|
if (cursorMask(mousePos)) {}
|
2021-01-14 20:47:06 -03:00
|
|
|
else if (cursorExit(mousePos)) {}
|
2021-01-10 11:18:55 -03:00
|
|
|
//
|
|
|
|
break;
|
|
|
|
|
2021-01-09 19:11:41 -03:00
|
|
|
default:
|
|
|
|
{}
|
2021-01-07 23:38:18 -03:00
|
|
|
|
2021-01-09 17:00:11 -03:00
|
|
|
}
|
2021-01-07 23:38:18 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Movies
|
|
|
|
if (_nextMovie != NULL) {
|
|
|
|
_videoDecoder = new Video::SmackerDecoder();
|
|
|
|
playVideo(*_nextMovie);
|
|
|
|
_nextMovie = NULL;
|
|
|
|
continue;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2021-01-16 15:30:00 -03:00
|
|
|
if (_nextVS != NULL) {
|
|
|
|
loadImage(*_nextVS, 160, 120, true);
|
|
|
|
_nextVS = NULL;
|
|
|
|
}
|
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
if (_videoDecoder) {
|
2021-01-14 18:48:30 -03:00
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
stopSound();
|
|
|
|
if (_videoDecoder->endOfVideo()) {
|
|
|
|
_videoDecoder->close();
|
|
|
|
delete _videoDecoder;
|
|
|
|
_videoDecoder = nullptr;
|
|
|
|
} else if (_videoDecoder->needsUpdate()) {
|
|
|
|
drawScreen();
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2021-01-09 19:11:41 -03:00
|
|
|
if (_nextSetting != NULL) {
|
2021-01-07 23:38:18 -03:00
|
|
|
debug("Executing %s", _nextSetting->c_str());
|
|
|
|
_exits.clear();
|
2021-01-08 22:42:34 -03:00
|
|
|
_masks.clear();
|
2021-01-12 20:49:12 -03:00
|
|
|
_loadGameMask = NULL;
|
|
|
|
_saveGameMask = NULL;
|
2021-01-07 23:38:18 -03:00
|
|
|
loadSetting(_nextSetting);
|
|
|
|
_nextSetting = NULL;
|
|
|
|
execute(prog);
|
2021-01-14 18:48:30 -03:00
|
|
|
changeCursor("default");
|
2021-01-07 23:38:18 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
g_system->updateScreen();
|
2021-01-09 17:00:11 -03:00
|
|
|
g_system->delayMillis(10);
|
2021-01-07 23:38:18 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
return Common::kNoError;
|
2020-12-24 16:10:32 -03:00
|
|
|
}
|
|
|
|
|
2021-01-10 11:18:55 -03:00
|
|
|
bool PrivateEngine::cursorExit(Common::Point mousePos) {
|
|
|
|
//debug("Mousepos %d %d", mousePos.x, mousePos.y);
|
|
|
|
mousePos = mousePos - *_origin;
|
|
|
|
if (mousePos.x < 0 || mousePos.y < 0)
|
|
|
|
return false;
|
2021-01-14 20:47:06 -03:00
|
|
|
|
|
|
|
int rs = 100000000;
|
|
|
|
int cs = 0;
|
2021-01-10 11:18:55 -03:00
|
|
|
ExitInfo e;
|
2021-01-14 20:47:06 -03:00
|
|
|
Common::String *cursor = NULL;
|
|
|
|
|
2021-01-10 11:18:55 -03:00
|
|
|
for (ExitList::iterator it = _exits.begin(); it != _exits.end(); ++it) {
|
|
|
|
e = *it;
|
2021-01-14 20:47:06 -03:00
|
|
|
cs = e.rect->width()*e.rect->height();
|
|
|
|
|
|
|
|
if (e.rect->contains(mousePos)) {
|
|
|
|
if (cs < rs && e.cursor != NULL) {
|
|
|
|
rs = cs;
|
|
|
|
cursor = e.cursor;
|
|
|
|
}
|
|
|
|
|
2021-01-10 11:18:55 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-14 20:47:06 -03:00
|
|
|
if (cursor != NULL) {
|
|
|
|
changeCursor(*cursor);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2021-01-10 11:18:55 -03:00
|
|
|
}
|
|
|
|
|
2021-01-16 15:30:00 -03:00
|
|
|
bool PrivateEngine::inMask(Graphics::ManagedSurface *surf, Common::Point mousePos) {
|
|
|
|
if (surf == NULL)
|
|
|
|
return false;
|
|
|
|
|
2021-01-10 11:18:55 -03:00
|
|
|
mousePos = mousePos - *_origin;
|
|
|
|
if (mousePos.x < 0 || mousePos.y < 0)
|
|
|
|
return false;
|
|
|
|
|
2021-01-16 15:30:00 -03:00
|
|
|
if (mousePos.x > surf->w || mousePos.y > surf->h)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return ( *((uint32*) surf->getBasePtr(mousePos.x, mousePos.y)) != _transparentColor);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool PrivateEngine::cursorMask(Common::Point mousePos) {
|
|
|
|
//debug("Mousepos %d %d", mousePos.x, mousePos.y);
|
2021-01-10 11:18:55 -03:00
|
|
|
MaskInfo m;
|
|
|
|
bool inside = false;
|
|
|
|
for (MaskList::iterator it = _masks.begin(); it != _masks.end(); ++it) {
|
|
|
|
m = *it;
|
|
|
|
|
2021-01-16 15:30:00 -03:00
|
|
|
if (inMask(m.surf, mousePos)) {
|
2021-01-10 11:18:55 -03:00
|
|
|
//debug("Inside!");
|
2021-01-16 15:30:00 -03:00
|
|
|
if (m.cursor != NULL) { // TODO: check this
|
2021-01-10 11:18:55 -03:00
|
|
|
inside = true;
|
2021-01-14 18:48:30 -03:00
|
|
|
//debug("Rendering cursor mask %s", m.cursor->c_str());
|
|
|
|
changeCursor(*m.cursor);
|
2021-01-10 11:18:55 -03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return inside;
|
|
|
|
}
|
|
|
|
|
2021-01-07 23:10:19 -03:00
|
|
|
void PrivateEngine::selectExit(Common::Point mousePos) {
|
2021-01-10 11:18:55 -03:00
|
|
|
//debug("Mousepos %d %d", mousePos.x, mousePos.y);
|
2021-01-09 19:11:41 -03:00
|
|
|
mousePos = mousePos - *_origin;
|
2021-01-10 11:18:55 -03:00
|
|
|
if (mousePos.x < 0 || mousePos.y < 0)
|
|
|
|
return;
|
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
Common::String *ns = NULL;
|
2021-01-08 22:42:34 -03:00
|
|
|
int rs = 100000000;
|
2021-01-07 23:38:18 -03:00
|
|
|
int cs = 0;
|
2021-01-08 22:42:34 -03:00
|
|
|
ExitInfo e;
|
|
|
|
for (ExitList::iterator it = _exits.begin(); it != _exits.end(); ++it) {
|
|
|
|
e = *it;
|
2021-01-09 19:11:41 -03:00
|
|
|
cs = e.rect->width()*e.rect->height();
|
2021-01-10 11:18:55 -03:00
|
|
|
//debug("Testing exit %s %d", e.nextSetting->c_str(), cs);
|
2021-01-07 23:38:18 -03:00
|
|
|
if (e.rect->contains(mousePos)) {
|
2021-01-14 18:48:30 -03:00
|
|
|
//debug("Inside! %d %d", cs, rs);
|
2021-01-08 22:42:34 -03:00
|
|
|
if (cs < rs && e.nextSetting != NULL) { // TODO: check this
|
2021-01-14 18:48:30 -03:00
|
|
|
//debug("Found Exit %s %d", e.nextSetting->c_str(), cs);
|
2021-01-07 23:38:18 -03:00
|
|
|
rs = cs;
|
|
|
|
ns = e.nextSetting;
|
|
|
|
}
|
|
|
|
|
2021-01-07 23:10:19 -03:00
|
|
|
}
|
2021-01-07 23:38:18 -03:00
|
|
|
}
|
2021-01-08 22:42:34 -03:00
|
|
|
if (ns != NULL) {
|
2021-01-14 18:48:30 -03:00
|
|
|
//debug("Exit selected %s", ns->c_str());
|
2021-01-07 23:38:18 -03:00
|
|
|
_nextSetting = ns;
|
2021-01-08 22:42:34 -03:00
|
|
|
}
|
2021-01-07 23:10:19 -03:00
|
|
|
}
|
|
|
|
|
2021-01-08 22:42:34 -03:00
|
|
|
void PrivateEngine::selectMask(Common::Point mousePos) {
|
|
|
|
Common::String *ns = NULL;
|
|
|
|
MaskInfo m;
|
|
|
|
for (MaskList::iterator it = _masks.begin(); it != _masks.end(); ++it) {
|
|
|
|
m = *it;
|
2021-01-09 19:11:41 -03:00
|
|
|
|
2021-01-10 11:18:55 -03:00
|
|
|
//debug("Testing mask %s", m.nextSetting->c_str());
|
2021-01-16 15:30:00 -03:00
|
|
|
if (inMask(m.surf, mousePos)) {
|
2021-01-14 18:48:30 -03:00
|
|
|
//debug("Inside!");
|
2021-01-08 22:42:34 -03:00
|
|
|
if (m.nextSetting != NULL) { // TODO: check this
|
2021-01-14 18:48:30 -03:00
|
|
|
//debug("Found Mask %s", m.nextSetting->c_str());
|
2021-01-08 22:42:34 -03:00
|
|
|
ns = m.nextSetting;
|
|
|
|
}
|
|
|
|
|
2021-01-10 15:38:10 -03:00
|
|
|
if (m.flag != NULL) { // TODO: check this
|
|
|
|
setSymbol(m.flag, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
2021-01-08 22:42:34 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (ns != NULL) {
|
2021-01-14 18:48:30 -03:00
|
|
|
//debug("Mask selected %s", ns->c_str());
|
2021-01-08 22:42:34 -03:00
|
|
|
_nextSetting = ns;
|
|
|
|
}
|
|
|
|
}
|
2021-01-07 23:10:19 -03:00
|
|
|
|
2021-01-16 15:30:00 -03:00
|
|
|
void PrivateEngine::selectAMRadioArea(Common::Point mousePos) {
|
|
|
|
if (_AMRadioArea == NULL)
|
2021-01-12 20:49:12 -03:00
|
|
|
return;
|
2021-01-16 15:30:00 -03:00
|
|
|
|
|
|
|
if (_AMRadio.empty())
|
2021-01-12 20:49:12 -03:00
|
|
|
return;
|
|
|
|
|
2021-01-16 15:30:00 -03:00
|
|
|
debug("AMRadio");
|
|
|
|
if (inMask(_AMRadioArea->surf, mousePos)) {
|
|
|
|
Common::String sound = *_AMRadioPrefix + _AMRadio.back() + ".wav";
|
|
|
|
playSound(sound.c_str(), 1);
|
|
|
|
_AMRadio.pop_back();
|
2021-01-12 20:49:12 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2021-01-16 15:30:00 -03:00
|
|
|
void PrivateEngine::selectPoliceRadioArea(Common::Point mousePos) {
|
|
|
|
if (_policeRadioArea == NULL)
|
|
|
|
return;
|
2021-01-12 20:49:12 -03:00
|
|
|
|
2021-01-16 15:30:00 -03:00
|
|
|
if (_policeRadio.empty())
|
|
|
|
return;
|
2021-01-12 20:49:12 -03:00
|
|
|
|
2021-01-16 15:30:00 -03:00
|
|
|
debug("PoliceRadio");
|
|
|
|
if (inMask(_policeRadioArea->surf, mousePos)) {
|
|
|
|
Common::String sound = *_policeRadioPrefix + _policeRadio.back() + ".wav";
|
|
|
|
playSound(sound.c_str(), 1);
|
|
|
|
_policeRadio.pop_back();
|
2021-01-12 20:49:12 -03:00
|
|
|
}
|
2021-01-16 15:30:00 -03:00
|
|
|
|
2021-01-12 20:49:12 -03:00
|
|
|
}
|
|
|
|
|
2021-01-16 15:30:00 -03:00
|
|
|
void PrivateEngine::selectLoadGame(Common::Point mousePos) {
|
|
|
|
if (_loadGameMask == NULL)
|
2021-01-12 20:49:12 -03:00
|
|
|
return;
|
|
|
|
|
2021-01-16 15:30:00 -03:00
|
|
|
if (inMask(_loadGameMask->surf, mousePos)) {
|
|
|
|
loadGameDialog();
|
2021-01-12 20:49:12 -03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-16 15:30:00 -03:00
|
|
|
void PrivateEngine::selectSaveGame(Common::Point mousePos) {
|
2021-01-12 20:49:12 -03:00
|
|
|
if (_saveGameMask == NULL)
|
2021-01-16 15:30:00 -03:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (inMask(_saveGameMask->surf, mousePos)) {
|
|
|
|
saveGameDialog();
|
2021-01-12 20:49:12 -03:00
|
|
|
}
|
|
|
|
|
2021-01-16 15:30:00 -03:00
|
|
|
}
|
2021-01-12 20:49:12 -03:00
|
|
|
|
2020-12-24 16:10:32 -03:00
|
|
|
bool PrivateEngine::hasFeature(EngineFeature f) const {
|
2021-01-07 23:38:18 -03:00
|
|
|
return
|
2021-01-10 15:38:10 -03:00
|
|
|
(f == kSupportsReturnToLauncher);
|
2020-12-24 16:10:32 -03:00
|
|
|
}
|
|
|
|
|
2021-01-16 15:30:00 -03:00
|
|
|
void PrivateEngine::restartGame() {
|
|
|
|
debug("restartGame");
|
|
|
|
|
|
|
|
for (VariableList::iterator it = variableList.begin(); it != variableList.end(); ++it) {
|
|
|
|
Private::Symbol *sym = variables.getVal(*it);
|
|
|
|
if (strcmp("kAlternateGame", sym->name->c_str()) != 0)
|
|
|
|
sym->u.val = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-24 16:10:32 -03:00
|
|
|
Common::Error PrivateEngine::loadGameStream(Common::SeekableReadStream *stream) {
|
2021-01-07 23:38:18 -03:00
|
|
|
Common::Serializer s(stream, nullptr);
|
2021-01-12 20:49:12 -03:00
|
|
|
debug("loadGameStream");
|
|
|
|
_nextSetting = new Common::String("kMainDesktop");
|
|
|
|
int val;
|
|
|
|
|
|
|
|
for (VariableList::iterator it = variableList.begin(); it != variableList.end(); ++it) {
|
|
|
|
s.syncAsUint32LE(val);
|
|
|
|
Private::Symbol *sym = variables.getVal(*it);
|
|
|
|
sym->u.val = val;
|
|
|
|
}
|
|
|
|
|
|
|
|
//syncGameStream(s);
|
2021-01-07 23:38:18 -03:00
|
|
|
return Common::kNoError;
|
2020-12-24 16:10:32 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
Common::Error PrivateEngine::saveGameStream(Common::WriteStream *stream, bool isAutosave) {
|
2021-01-12 20:49:12 -03:00
|
|
|
debug("saveGameStream %d", isAutosave);
|
|
|
|
if (isAutosave)
|
|
|
|
return Common::kNoError;
|
2021-01-14 18:48:30 -03:00
|
|
|
|
2021-01-12 20:49:12 -03:00
|
|
|
for (VariableList::iterator it = variableList.begin(); it != variableList.end(); ++it) {
|
|
|
|
Private::Symbol *sym = variables.getVal(*it);
|
|
|
|
stream->writeUint32LE(sym->u.val);
|
|
|
|
}
|
2021-01-07 23:38:18 -03:00
|
|
|
return Common::kNoError;
|
2020-12-24 16:10:32 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
void PrivateEngine::syncGameStream(Common::Serializer &s) {
|
2021-01-12 20:49:12 -03:00
|
|
|
debug("syncGameStream");
|
2021-01-07 23:38:18 -03:00
|
|
|
// Use methods of Serializer to save/load fields
|
2021-01-12 20:49:12 -03:00
|
|
|
//int dummy = 0;
|
|
|
|
//s.syncString(*_currentSetting);
|
2020-12-24 16:10:32 -03:00
|
|
|
}
|
|
|
|
|
2021-01-07 23:32:17 -03:00
|
|
|
Common::String PrivateEngine::convertPath(Common::String name) {
|
2021-01-07 23:38:18 -03:00
|
|
|
Common::String path(name);
|
|
|
|
Common::String s1("\\");
|
|
|
|
Common::String s2("/");
|
2021-01-07 23:32:17 -03:00
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
while (path.contains(s1))
|
|
|
|
Common::replace(path, s1, s2);
|
2021-01-07 23:32:17 -03:00
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
s1 = Common::String("\"");
|
|
|
|
s2 = Common::String("");
|
2021-01-07 23:32:17 -03:00
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
Common::replace(path, s1, s2);
|
|
|
|
Common::replace(path, s1, s2);
|
|
|
|
|
|
|
|
path.toLowercase();
|
|
|
|
return path;
|
2021-01-07 23:32:17 -03:00
|
|
|
}
|
|
|
|
|
2021-01-14 20:47:06 -03:00
|
|
|
void PrivateEngine::playSound(const Common::String &name, uint loops) {
|
2021-01-07 23:38:18 -03:00
|
|
|
debugC(1, kPrivateDebugExample, "%s : %s", __FUNCTION__, name.c_str());
|
2021-01-02 10:22:07 -03:00
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
Common::File *file = new Common::File();
|
|
|
|
Common::String path = convertPath(name);
|
2021-01-02 14:19:23 -03:00
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
if (!file->open(path))
|
|
|
|
error("unable to find sound file %s", path.c_str());
|
2020-12-24 19:02:12 -03:00
|
|
|
|
2021-01-09 16:11:30 -03:00
|
|
|
Audio::LoopingAudioStream *stream;
|
2021-01-14 20:47:06 -03:00
|
|
|
stream = new Audio::LoopingAudioStream(Audio::makeWAVStream(file, DisposeAfterUse::YES), loops);
|
2021-01-07 23:38:18 -03:00
|
|
|
stopSound();
|
|
|
|
_mixer->playStream(Audio::Mixer::kSFXSoundType, &_soundHandle, stream, -1, Audio::Mixer::kMaxChannelVolume);
|
2020-12-24 19:02:12 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
void PrivateEngine::playVideo(const Common::String &name) {
|
2021-01-08 22:42:34 -03:00
|
|
|
debug("%s : %s", __FUNCTION__, name.c_str());
|
2021-01-07 23:38:18 -03:00
|
|
|
Common::File *file = new Common::File();
|
|
|
|
Common::String path = convertPath(name);
|
2021-01-04 08:14:40 -03:00
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
if (!file->open(path))
|
|
|
|
error("unable to find video file %s", path.c_str());
|
2020-12-24 19:02:12 -03:00
|
|
|
|
2021-01-07 23:38:18 -03:00
|
|
|
if (!_videoDecoder->loadStream(file))
|
|
|
|
error("unable to load video %s", path.c_str());
|
|
|
|
_videoDecoder->start();
|
2020-12-24 19:02:12 -03:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2021-01-09 16:11:30 -03:00
|
|
|
void PrivateEngine::skipVideo() {
|
|
|
|
_videoDecoder->close();
|
|
|
|
delete _videoDecoder;
|
|
|
|
_videoDecoder = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-12-24 19:02:12 -03:00
|
|
|
void PrivateEngine::stopSound() {
|
2021-01-07 23:38:18 -03:00
|
|
|
debugC(3, kPrivateDebugExample, "%s", __FUNCTION__);
|
|
|
|
if (_mixer->isSoundHandleActive(_soundHandle))
|
|
|
|
_mixer->stopHandle(_soundHandle);
|
2020-12-24 19:02:12 -03:00
|
|
|
}
|
|
|
|
|
2021-01-09 16:11:30 -03:00
|
|
|
void PrivateEngine::loadImage(const Common::String &name, int x, int y, bool drawn) {
|
2021-01-07 23:38:18 -03:00
|
|
|
debugC(1, kPrivateDebugExample, "%s : %s", __FUNCTION__, name.c_str());
|
|
|
|
Common::File file;
|
|
|
|
Common::String path = convertPath(name);
|
|
|
|
if (!file.open(path))
|
|
|
|
error("unable to load image %s", path.c_str());
|
|
|
|
|
|
|
|
_image->loadStream(file);
|
2021-01-09 16:11:30 -03:00
|
|
|
_compositeSurface->transBlitFrom(*_image->getSurface()->convertTo(_pixelFormat, _image->getPalette()), *_origin + Common::Point(x,y), _transparentColor);
|
2021-01-07 23:38:18 -03:00
|
|
|
drawScreen();
|
2021-01-02 14:19:23 -03:00
|
|
|
}
|
|
|
|
|
2021-01-14 18:48:30 -03:00
|
|
|
void PrivateEngine::drawScreenFrame(Graphics::Surface *screen) {
|
2021-01-09 16:11:30 -03:00
|
|
|
Common::String path = convertPath(*_frame);
|
2021-01-14 18:48:30 -03:00
|
|
|
Common::File file;
|
|
|
|
assert(file.open(path));
|
2021-01-09 16:11:30 -03:00
|
|
|
_image->loadStream(file);
|
2021-01-14 18:48:30 -03:00
|
|
|
screen->copyRectToSurface(*_image->getSurface()->convertTo(_pixelFormat, _image->getPalette()), 0, 0, Common::Rect(0, 0, _screenW, _screenH));
|
2021-01-09 16:11:30 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-01-08 22:42:34 -03:00
|
|
|
Graphics::ManagedSurface *PrivateEngine::loadMask(const Common::String &name, int x, int y, bool drawn) {
|
|
|
|
debugC(1, kPrivateDebugExample, "%s : %s", __FUNCTION__, name.c_str());
|
|
|
|
Common::File file;
|
|
|
|
Common::String path = convertPath(name);
|
|
|
|
if (!file.open(path))
|
|
|
|
error("unable to load mask %s", path.c_str());
|
|
|
|
|
|
|
|
_image->loadStream(file);
|
|
|
|
Graphics::ManagedSurface *surf = new Graphics::ManagedSurface();
|
|
|
|
surf->create(_screenW, _screenH, _pixelFormat);
|
|
|
|
surf->transBlitFrom(*_image->getSurface()->convertTo(_pixelFormat, _image->getPalette()), Common::Point(x,y));
|
|
|
|
|
|
|
|
if (drawn) {
|
2021-01-10 11:18:55 -03:00
|
|
|
_compositeSurface->transBlitFrom(surf->rawSurface(), *_origin + Common::Point(x,y), _transparentColor);
|
2021-01-08 22:42:34 -03:00
|
|
|
drawScreen();
|
|
|
|
}
|
|
|
|
|
2021-01-09 19:11:41 -03:00
|
|
|
return surf;
|
2021-01-08 22:42:34 -03:00
|
|
|
}
|
|
|
|
|
2021-01-10 20:17:27 -03:00
|
|
|
void PrivateEngine::drawScreen() {
|
2021-01-10 11:18:55 -03:00
|
|
|
Graphics::Surface *screen = g_system->lockScreen();
|
|
|
|
Graphics::ManagedSurface *surface = _compositeSurface;
|
|
|
|
int w = surface->w;
|
|
|
|
int h = surface->h;
|
|
|
|
|
|
|
|
if (_videoDecoder) {
|
|
|
|
Graphics::Surface *frame = new Graphics::Surface;
|
|
|
|
frame->create(_screenW, _screenH, _pixelFormat);
|
|
|
|
frame->copyFrom(*_videoDecoder->decodeNextFrame());
|
|
|
|
const Common::Point o(_origin->x, _origin->y);
|
|
|
|
surface->transBlitFrom(*frame->convertTo(_pixelFormat, _videoDecoder->getPalette()), o);
|
|
|
|
}
|
2021-01-08 22:42:34 -03:00
|
|
|
|
2021-01-10 11:18:55 -03:00
|
|
|
assert(w == _screenW && h == _screenH);
|
2021-01-08 22:42:34 -03:00
|
|
|
|
2021-01-14 18:48:30 -03:00
|
|
|
if (_mode == 1) {
|
|
|
|
drawScreenFrame(screen);
|
|
|
|
}
|
|
|
|
screen->copyRectToSurface(*surface, _origin->x, _origin->y, Common::Rect(_origin->x, _origin->y, _screenW - _origin->x, _screenH - _origin->y));
|
|
|
|
//screen->copyRectToSurface(*surface, _origin.x, _origin.y, Common::Rect(0, 0, _screenW, _screenH));
|
2021-01-10 11:18:55 -03:00
|
|
|
g_system->unlockScreen();
|
|
|
|
//if (_image->getPalette() != nullptr)
|
|
|
|
// g_system->getPaletteManager()->setPalette(_image->getPalette(), _image->getPaletteStartIndex(), _image->getPaletteColorCount());
|
|
|
|
//if (_image->getPalette() != nullptr)
|
|
|
|
// g_system->getPaletteManager()->setPalette(_image->getPalette(), 0, 256);
|
|
|
|
//g_system->getPaletteManager()->setPalette(_videoDecoder->getPalette(), 0, 256);
|
|
|
|
g_system->updateScreen();
|
2021-01-02 14:19:23 -03:00
|
|
|
|
2020-12-24 19:02:12 -03:00
|
|
|
}
|
|
|
|
|
2021-01-10 15:38:10 -03:00
|
|
|
bool PrivateEngine::getRandomBool(uint p) {
|
|
|
|
uint r = _rnd->getRandomNumber(100);
|
2021-01-10 20:17:27 -03:00
|
|
|
return (r <= p);
|
2021-01-10 15:38:10 -03:00
|
|
|
}
|
2020-12-24 19:02:12 -03:00
|
|
|
|
|
|
|
|
2020-12-24 16:10:32 -03:00
|
|
|
} // End of namespace Private
|