scummvm/engines/pink/pink.cpp
2023-12-24 13:19:25 +01:00

328 lines
9.2 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
#include "common/debug-channels.h"
#include "common/formats/winexe_pe.h"
#include "common/config-manager.h"
#include "common/compression/installshield_cab.h"
#include "engines/advancedDetector.h"
#include "engines/util.h"
#include "graphics/cursorman.h"
#include "graphics/thumbnail.h"
#include "graphics/surface.h"
#include "graphics/wincursor.h"
#include "pink/pink.h"
#include "pink/console.h"
#include "pink/screen.h"
#include "pink/objects/module.h"
#include "pink/objects/actors/lead_actor.h"
namespace Pink {
Graphics::PaletteLookup *g_paletteLookup;
PinkEngine::PinkEngine(OSystem *system, const ADGameDescription *desc)
: Engine(system), _rnd("pink"), _exeResources(nullptr),
_desc(desc), _bro(nullptr), _menu(nullptr), _actor(nullptr),
_module(nullptr), _screen(nullptr), _pdaMgr(this) {
const Common::FSNode gameDataDir(ConfMan.getPath("path"));
SearchMan.addSubDirectoryMatching(gameDataDir, "install");
g_paletteLookup = new Graphics::PaletteLookup;
_isPeril = !strcmp(_desc->gameId, kPeril);
_isPerilDemo = _isPeril && (_desc->flags & ADGF_DEMO);
}
PinkEngine::~PinkEngine() {
delete _exeResources;
delete _bro;
_pdaMgr.close();
for (uint i = 0; i < _modules.size(); ++i) {
delete _modules[i];
}
for (uint j = 0; j < _cursors.size(); ++j) {
delete _cursors[j];
}
delete _screen;
delete g_paletteLookup;
}
Common::Error PinkEngine::init() {
debugC(10, kPinkDebugGeneral, "PinkEngine init");
initGraphics(640, 480);
_exeResources = new Common::PEResources();
Common::Path fileName = isPeril() ? "pptp.exe" : "hpp.exe";
if ((_desc->flags & GF_COMPRESSED) && isPeril()) {
fileName = "pptp.ex_";
Common::Archive *cabinet = Common::makeInstallShieldArchive("data");
if (!cabinet)
error("Failed to open the InstallShield cabinet");
SearchMan.add("data1.cab", cabinet);
}
if (!_exeResources->loadFromEXE(fileName)) {
return Common::kNoGameDataFoundError;
}
setDebugger(new Console(this));
_screen = new Screen(this);
initMenu();
Common::Path orbName;
Common::Path broName;
if (isPeril()) {
orbName = "PPTP.ORB";
broName = "PPTP.BRO";
_bro = new BroFile;
} else {
orbName = "HPP.ORB";
}
if (!_orb.open(orbName))
return Common::kNoGameDataFoundError;
if (_bro) {
if (!_bro->open(broName))
return Common::kNoGameDataFoundError;
if (_orb.getTimestamp() != _bro->getTimestamp()) {
warning("ORB and BRO timestamp mismatch. %x != %x", _orb.getTimestamp(), _bro->getTimestamp());
//return Common::kNoGameDataFoundError;
}
}
if (!loadCursors())
return Common::kNoGameDataFoundError;
setCursor(kLoadingCursor);
_orb.loadGame(this);
debugC(6, kPinkDebugGeneral, "Modules are loaded");
syncSoundSettings();
if (ConfMan.hasKey("save_slot"))
loadGameState(ConfMan.getInt("save_slot"));
else
initModule(_modules[0]->getName(), "", nullptr);
return Common::kNoError;
}
Common::Error Pink::PinkEngine::run() {
Common::Error error = init();
if (error.getCode() != Common::kNoError)
return error;
while (!shouldQuit()) {
Common::Event event;
while (_eventMan->pollEvent(event)) {
if (_screen->processEvent(event))
continue;
switch (event.type) {
case Common::EVENT_QUIT:
case Common::EVENT_RETURN_TO_LAUNCHER:
return Common::kNoError;
case Common::EVENT_MOUSEMOVE:
_actor->onMouseMove(event.mouse);
break;
case Common::EVENT_LBUTTONDOWN:
_actor->onLeftButtonClick(event.mouse);
break;
case Common::EVENT_LBUTTONUP:
_actor->onLeftButtonUp();
break;
case Common::EVENT_RBUTTONDOWN:
if (isPeril())
_actor->onRightButtonClick(event.mouse);
break;
case Common::EVENT_KEYDOWN:
_actor->onKeyboardButtonClick(event.kbd.keycode);
break;
default:
break;
}
}
_actor->update();
_screen->update();
_system->delayMillis(10);
}
return Common::kNoError;
}
void PinkEngine::load(Archive &archive) {
archive.skipString();
archive.skipString();
_modules.deserialize(archive);
}
void PinkEngine::initModule(const Common::String moduleName, const Common::String pageName, Archive *saveFile) {
if (_module)
removeModule();
if (moduleName == _modules[0]->getName()) {
// new game
_variables.clear();
debugC(6, kPinkDebugGeneral, "Global Game Variables cleared");
}
addModule(moduleName);
if (saveFile)
_module->loadState(*saveFile);
debugC(6, kPinkDebugGeneral, "Module added");
_module->init(saveFile != nullptr, pageName);
}
void PinkEngine::changeScene() {
setCursor(kLoadingCursor);
_screen->clear();
if (!_nextModule.empty() && _nextModule != _module->getName())
initModule(_nextModule, _nextPage, nullptr);
else
_module->changePage(_nextPage);
}
void PinkEngine::addModule(const Common::String &moduleName) {
_module = new Module(this, moduleName);
_orb.loadObject(_module, _module->getName());
for (uint i = 0; i < _modules.size(); ++i) {
if (_modules[i]->getName() == moduleName) {
delete _modules[i];
_modules[i] = _module;
break;
}
}
}
void PinkEngine::removeModule() {
for (uint i = 0; i < _modules.size(); ++i) {
if (_module == _modules[i]) {
_pdaMgr.close();
_modules[i] = new ModuleProxy(Common::String(_module->getName()));
delete _module;
_module = nullptr;
break;
}
}
}
void PinkEngine::setVariable(Common::String &variable, Common::String &value) {
_variables[variable] = value;
}
bool PinkEngine::checkValueOfVariable(const Common::String &variable, const Common::String &value) const {
if (!_variables.contains(variable))
return value == kUndefinedValue;
return _variables[variable] == value;
}
bool PinkEngine::loadCursors() {
bool isPokus = !isPeril();
_cursors.reserve(kCursorsCount);
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPokusLoadingCursorID));
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPokusClickableFirstCursorID));
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPokusClickableSecondCursorID));
if (isPokus) {
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPokusClickableThirdCursorID));
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPokusNotClickableCursorID));
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPokusHoldingItemCursorID));
} else {
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPerilClickableThirdCursorID));
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPerilNotClickableCursorID));
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPerilHoldingItemCursorID));
}
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPokusPDADefaultCursorID));
if (isPokus) {
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPokusPDAClickableFirstFrameCursorID));
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPokusPDAClickableSecondFrameCursorID));
} else {
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPerilPDAClickableFirstFrameCursorID));
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPerilPDAClickableSecondFrameCursorID));
}
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPokusExitLeftCursorID));
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPokusExitRightCursorID));
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPokusExitForwardCursorID));
if (isPokus)
_cursors.push_back(Graphics::WinCursorGroup::createCursorGroup(_exeResources, kPokusExitDownCursorID));
return true;
}
void PinkEngine::setCursor(uint cursorIndex) {
CursorMan.replaceCursor(_cursors[cursorIndex]->cursors[0].cursor);
CursorMan.showMouse(true);
}
bool PinkEngine::canLoadGameStateCurrently(Common::U32String *msg) {
return true;
}
bool PinkEngine::canSaveGameStateCurrently(Common::U32String *msg) {
return true;
}
bool PinkEngine::hasFeature(Engine::EngineFeature f) const {
return
f == kSupportsReturnToLauncher ||
f == kSupportsLoadingDuringRuntime ||
f == kSupportsSavingDuringRuntime ||
f == kSupportsChangingOptionsDuringRuntime;
}
void PinkEngine::pauseEngineIntern(bool pause) {
Engine::pauseEngineIntern(pause);
_screen->pause(pause);
}
bool PinkEngine::isPeril() const {
return _isPeril;
}
bool PinkEngine::isPerilDemo() const {
return _isPerilDemo;
}
}