mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-05 17:20:30 +00:00
297 lines
8.5 KiB
C++
297 lines
8.5 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/winexe_pe.h"
|
|
#include "common/config-manager.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/director.h"
|
|
#include "pink/objects/module.h"
|
|
#include "pink/objects/actors/lead_actor.h"
|
|
|
|
namespace Pink {
|
|
|
|
PinkEngine::PinkEngine(OSystem *system, const ADGameDescription *desc)
|
|
: Engine(system), _rnd("pink"), _exeResources(nullptr),
|
|
_desc(desc), _bro(nullptr), _menu(nullptr), _actor(nullptr),
|
|
_module(nullptr), _director(nullptr), _pdaMgr(this) {
|
|
|
|
const Common::FSNode gameDataDir(ConfMan.get("path"));
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "install");
|
|
}
|
|
|
|
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 _director;
|
|
}
|
|
|
|
Common::Error PinkEngine::init() {
|
|
debugC(10, kPinkDebugGeneral, "PinkEngine init");
|
|
initGraphics(640, 480);
|
|
|
|
_exeResources = new Common::PEResources();
|
|
Common::String fileName = isPeril() ? "pptp.exe" : "hpp.exe";
|
|
if (!_exeResources->loadFromEXE(fileName)) {
|
|
return Common::kNoGameDataFoundError;
|
|
}
|
|
|
|
setDebugger(new Console(this));
|
|
_director = new Director(this);
|
|
|
|
initMenu();
|
|
|
|
Common::String orbName;
|
|
Common::String 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 (_director->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();
|
|
_director->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();
|
|
|
|
addModule(moduleName);
|
|
if (saveFile)
|
|
_module->loadState(*saveFile);
|
|
|
|
debugC(6, kPinkDebugGeneral, "Module added");
|
|
|
|
_module->init(saveFile ? kLoadingSave : kLoadingNewGame, pageName);
|
|
}
|
|
|
|
void PinkEngine::changeScene() {
|
|
setCursor(kLoadingCursor);
|
|
_director->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(_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() {
|
|
return true;
|
|
}
|
|
|
|
bool PinkEngine::canSaveGameStateCurrently() {
|
|
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);
|
|
_director->pause(pause);
|
|
}
|
|
|
|
bool PinkEngine::isPeril() const {
|
|
return !strcmp(_desc->gameId, kPeril);
|
|
}
|
|
|
|
}
|