scummvm/engines/macventure/macventure.cpp
2022-09-04 20:30:02 +02:00

1182 lines
29 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/>.
*
*/
/*
* Based on
* WebVenture (c) 2010, Sean Kasun
* https://github.com/mrkite/webventure, http://seancode.com/webventure/
*
* Used with explicit permission from the author
*/
#include "common/system.h"
#include "common/debug-channels.h"
#include "common/debug.h"
#include "common/error.h"
#include "common/config-manager.h"
#include "engines/util.h"
#include "macventure/macventure.h"
// To move
#include "common/file.h"
namespace MacVenture {
// HACK, see below
void toASCII(Common::String &str) {
debugC(3, kMVDebugMain, "toASCII: %s", str.c_str());
Common::String::iterator it = str.begin();
for (; it != str.end(); it++) {
if (*it == '\216') {
str.replace(it, it + 1, "e");
}
if (*it == '\210') {
str.replace(it, it + 1, "a");
}
}
}
enum {
kMaxMenuTitleLength = 30
};
MacVentureEngine::MacVentureEngine(OSystem *syst, const ADGameDescription *gameDesc) : Engine(syst) {
_gameDescription = gameDesc;
_rnd = new Common::RandomSource("macventure");
_resourceManager = nullptr;
_globalSettings = nullptr;
_gui = nullptr;
_world = nullptr;
_scriptEngine = nullptr;
_filenames = nullptr;
_decodingDirectArticles = nullptr;
_decodingNamingArticles = nullptr;
_decodingIndirectArticles = nullptr;
_textHuffman = nullptr;
_soundManager = nullptr;
_dataBundle = nullptr;
debug("MacVenture::MacVentureEngine()");
}
MacVentureEngine::~MacVentureEngine() {
debug("MacVenture::~MacVentureEngine()");
if (_rnd)
delete _rnd;
if (_resourceManager)
delete _resourceManager;
if (_globalSettings)
delete _globalSettings;
if (_gui)
delete _gui;
if (_world)
delete _world;
if (_scriptEngine)
delete _scriptEngine;
if (_filenames)
delete _filenames;
if (_decodingDirectArticles)
delete _decodingDirectArticles;
if (_decodingNamingArticles)
delete _decodingNamingArticles;
if (_decodingIndirectArticles)
delete _decodingIndirectArticles;
if (_textHuffman)
delete _textHuffman;
if (_soundManager)
delete _soundManager;
if (_dataBundle)
delete _dataBundle;
}
Common::Error MacVentureEngine::run() {
debug("MacVenture::MacVentureEngine::init()");
initGraphics(kScreenWidth, kScreenHeight);
setInitialFlags();
setDebugger(new Console(this));
// Additional setup.
debug("MacVentureEngine::init");
_resourceManager = new Common::MacResManager();
if (!_resourceManager->open(getGameFileName()))
error("ENGINE: Could not open %s as a resource fork", getGameFileName());
// Engine-wide loading
if (!loadGlobalSettings())
error("ENGINE: Could not load the engine settings");
_oldTextEncoding = !loadTextHuffman();
_filenames = new StringTable(this, _resourceManager, kFilenamesStringTableID);
_decodingDirectArticles = new StringTable(this, _resourceManager, kCommonArticlesStringTableID);
_decodingNamingArticles = new StringTable(this, _resourceManager, kNamingArticlesStringTableID);
_decodingIndirectArticles = new StringTable(this, _resourceManager, kIndirectArticlesStringTableID);
loadDataBundle();
// Big class instantiation
_gui = new Gui(this, _resourceManager);
_world = new World(this, _resourceManager);
_scriptEngine = new ScriptEngine(this, _world);
_soundManager = new SoundManager(this, _mixer);
int directSaveSlotLoading = ConfMan.getInt("save_slot");
if (directSaveSlotLoading >= 0) {
if (loadGameState(directSaveSlotLoading).getCode() != Common::kNoError) {
error("ENGINE: Could not load game from slot '%d'", directSaveSlotLoading);
}
} else {
setNewGameState();
}
selectControl(kStartOrResume);
_gui->addChild(kSelfWindow, 1);
_gui->updateWindow(kSelfWindow, false);
while (_gameState != kGameStateQuitting) {
processEvents();
if (_gameState != kGameStateQuitting && !_gui->isDialogOpen()) {
if (_prepared) {
_prepared = false;
if (!_halted)
updateState(false);
if (_cmdReady || _halted) {
_halted = false;
if (runScriptEngine()) {
_halted = true;
_paused = true;
} else {
_paused = false;
updateState(true);
updateControls();
updateExits();
}
}
if (_gameState == kGameStateWinnig || _gameState == kGameStateLosing) {
endGame();
}
}
}
refreshScreen();
}
return Common::kNoError;
}
void MacVentureEngine::refreshScreen() {
_gui->draw();
g_system->updateScreen();
g_system->delayMillis(50);
}
void MacVentureEngine::newGame() {
_world->startNewGame();
reset();
setInitialFlags();
setNewGameState();
}
void MacVentureEngine::setInitialFlags() {
_paused = false;
_halted = false;
_cmdReady = false;
_haltedAtEnd = false;
_haltedInSelection = false;
_clickToContinue = true;
_gameState = kGameStateInit;
_destObject = 0;
_prepared = true;
}
void MacVentureEngine::setNewGameState() {
_cmdReady = true;
ObjID playerParent = _world->getObjAttr(1, kAttrParentObject);
_currentSelection.push_back(playerParent);// Push the parent of the player
_world->setObjAttr(playerParent, kAttrContainerOpen, 1);
}
void MacVentureEngine::reset() {
resetInternals();
resetGui();
}
void MacVentureEngine::resetInternals() {
_scriptEngine->reset();
_currentSelection.clear();
_objQueue.clear();
_textQueue.clear();
}
void MacVentureEngine::resetGui() {
_gui->reloadInternals();
_gui->updateWindowInfo(kMainGameWindow, getParent(1), _world->getChildren(getParent(1), true));
// HACK! should update all inventories
_gui->ensureInventoryOpen(kInventoryStart, 1);
_gui->updateWindowInfo(kInventoryStart, 1, _world->getChildren(1, true));
updateControls();
updateExits();
refreshScreen();
}
void MacVentureEngine::requestQuit() {
// TODO: Display save game dialog and such
_gameState = kGameStateQuitting;
}
void MacVentureEngine::requestUnpause() {
_paused = false;
_gameState = kGameStatePlaying;
}
void MacVentureEngine::selectControl(ControlAction id) {
debugC(2, kMVDebugMain, "Select control %x", id);
if (id == kClickToContinue) {
_clickToContinue = false;
_paused = true;
return;
}
_selectedControl = id;
refreshReady();
}
void MacVentureEngine::refreshReady() {
switch (getInvolvedObjects()) {
case 0: // No selected object
_cmdReady = true;
break;
case 1: // We have some selected object
_cmdReady = _currentSelection.size() != 0;
break;
case 2:
if (_destObject > 0) // We have a destination selected
_cmdReady = true;
break;
default:
break;
}
}
void MacVentureEngine::preparedToRun() {
_prepared = true;
}
void MacVentureEngine::gameChanged() {
_gameChanged = true;
}
void MacVentureEngine::winGame() {
_gui->showPrebuiltDialog(kWinGameDialog);
_gameState = kGameStateWinnig;
}
void MacVentureEngine::loseGame() {
_gui->showPrebuiltDialog(kLoseGameDialog);
_paused = true;
//_gameState = kGameStateLosing;
}
void MacVentureEngine::clickToContinue() {
_clickToContinue = true;
}
void MacVentureEngine::enqueueObject(ObjectQueueID type, ObjID objID, ObjID target) {
QueuedObject obj;
obj.id = type;
if (type == kUpdateObject && isObjEnqueued(objID)) {
return;
}
if (type == kUpdateWindow) {
obj.target = target;
}
if (type != kHightlightExits) {
obj.object = objID;
obj.parent = _world->getObjAttr(objID, kAttrParentObject);
obj.x = _world->getObjAttr(objID, kAttrPosX);
obj.y = _world->getObjAttr(objID, kAttrPosY);
obj.exitx = _world->getObjAttr(objID, kAttrExitX);
obj.exity = _world->getObjAttr(objID, kAttrExitY);
obj.hidden = _world->getObjAttr(objID, kAttrHiddenExit);
obj.offscreen = _world->getObjAttr(objID, kAttrInvisible);
obj.invisible = _world->getObjAttr(objID, kAttrUnclickable);
}
_objQueue.push_back(obj);
}
void MacVentureEngine::enqueueText(TextQueueID type, ObjID target, ObjID source, ObjID text) {
QueuedText newText;
newText.id = type;
newText.destination = target;
newText.source = source;
newText.asset = text;
_textQueue.push_back(newText);
}
void MacVentureEngine::enqueueSound(SoundQueueID type, ObjID target) {
QueuedSound newSound;
newSound.id = type;
newSound.reference = target;
_soundQueue.push_back(newSound);
}
void MacVentureEngine::handleObjectSelect(ObjID objID, WindowReference win, bool shiftPressed, bool isDoubleClick) {
if (win == kExitsWindow) {
win = kMainGameWindow;
}
const WindowData &windata = _gui->getWindowData(win);
if (shiftPressed) {
// TODO: Implement shift functionality.
} else {
if (_selectedControl && _currentSelection.size() > 0 && getInvolvedObjects() > 1) {
if (objID == 0) {
selectPrimaryObject(windata.objRef);
} else {
selectPrimaryObject(objID);
}
preparedToRun();
} else {
if (objID == 0) {
unselectAll();
objID = win;
}
if (objID > 0) {
int currentObjectIndex = findObjectInArray(objID, _currentSelection);
if (currentObjectIndex >= 0)
unselectAll();
if (isDoubleClick) {
selectObject(objID);
_destObject = objID;
setDeltaPoint(Common::Point(0, 0));
if (!_cmdReady) {
selectControl(kActivateObject);
_cmdReady = true;
}
} else {
selectObject(objID);
if (getInvolvedObjects() == 1)
_cmdReady = true;
}
preparedToRun();
}
}
}
}
void MacVentureEngine::handleObjectDrop(ObjID objID, Common::Point delta, ObjID newParent) {
_destObject = newParent;
setDeltaPoint(delta);
selectControl(kMoveObject);
refreshReady();
preparedToRun();
}
void MacVentureEngine::setDeltaPoint(Common::Point newPos) {
debugC(4, kMVDebugMain, "Update delta: Old(%d, %d), New(%d, %d)",
_deltaPoint.x, _deltaPoint.y,
newPos.x, newPos.y);
_deltaPoint = newPos;
}
void MacVentureEngine::focusObjWin(ObjID objID) {
_gui->bringToFront(getObjWindow(objID));
}
void MacVentureEngine::updateWindow(WindowReference winID) {
_gui->updateWindow(winID, true);
}
bool MacVentureEngine::showTextEntry(ObjID text, ObjID srcObj, ObjID destObj) {
debugC(3, kMVDebugMain, "Showing speech dialog, asset %d from %d to %d", text, srcObj, destObj);
_gui->getTextFromUser();
_prepared = false;
warning("Show text entry: not fully tested");
return true;
}
void MacVentureEngine::setTextInput(const Common::String &content) {
_prepared = true;
_userInput = content;
_clickToContinue = false;
}
Common::String MacVentureEngine::getUserInput() {
return _userInput;
}
Common::String MacVentureEngine::getStartGameFileName() {
Common::SeekableReadStream *res;
res = _resourceManager->getResource(MKTAG('S', 'T', 'R', ' '), kStartGameFilenameID);
if (!res)
return "";
byte length = res->readByte();
char *fileName = new char[length + 1];
res->read(fileName, length);
fileName[length] = '\0';
Common::String result = Common::String(fileName, length);
// HACK, see definition of toASCII
toASCII(result);
delete[] fileName;
delete res;
return result;
}
const GlobalSettings& MacVentureEngine::getGlobalSettings() const {
return *_globalSettings;
}
// Private engine methods
void MacVentureEngine::processEvents() {
Common::Event event;
while (_eventMan->pollEvent(event)) {
if (_gui->processEvent(event))
continue;
switch (event.type) {
case Common::EVENT_QUIT:
case Common::EVENT_RETURN_TO_LAUNCHER:
_gameState = kGameStateQuitting;
break;
default:
break;
}
}
}
bool MacVenture::MacVentureEngine::runScriptEngine() {
debugC(3, kMVDebugMain, "Running script engine");
if (_haltedAtEnd) {
_haltedAtEnd = false;
if (_scriptEngine->resume(false)) {
_haltedAtEnd = true;
return true;
}
return false;
}
if (_haltedInSelection) {
_haltedInSelection = false;
if (_scriptEngine->resume(false)) {
_haltedInSelection = true;
return true;
}
updateState(true);
}
while (!_currentSelection.empty()) {
ObjID obj = _currentSelection.front();
_currentSelection.remove_at(0);
if (isGameRunning() && _world->isObjActive(obj)) {
if (_scriptEngine->runControl(_selectedControl, obj, _destObject, _deltaPoint)) {
_haltedInSelection = true;
return true;
}
updateState(true);
}
}
if (_selectedControl == 1) {
_gameChanged = false;
} else if (isGameRunning()) {
if (_scriptEngine->runControl(kTick, _selectedControl, _destObject, _deltaPoint)) {
_haltedAtEnd = true;
return true;
}
}
return false;
}
void MacVentureEngine::endGame() {
requestQuit();
}
void MacVentureEngine::updateState(bool pause) {
_prepared = false;
runObjQueue();
printTexts();
playSounds(pause);
}
void MacVentureEngine::revert() {
_gui->invertWindowColors(kMainGameWindow);
preparedToRun();
}
void MacVentureEngine::runObjQueue() {
while (!_objQueue.empty()) {
uint32 biggest = 0;
uint32 index = 0;
uint32 temp;
for (uint i = 0; i < _objQueue.size(); i++) {
temp = _objQueue[i].id;
if (temp > biggest) {
biggest = temp;
index = i;
}
}
QueuedObject obj = _objQueue[index];
_objQueue.remove_at(index);
switch (obj.id) {
case 0x2:
focusObjectWindow(obj.object);
break;
case 0x3:
openObject(obj.object);
break;
case 0x4:
closeObject(obj.object);
break;
case 0x7:
checkObject(obj);
break;
case 0x8:
reflectSwap(obj.object, obj.target);
break;
case 0xc:
_world->setObjAttr(_gui->getWindowData(kMainGameWindow).refcon, kAttrContainerOpen, 0);
_world->setObjAttr(_world->getObjAttr(1, kAttrParentObject), kAttrContainerOpen, 1);
break;
case 0xd:
toggleExits();
break;
case 0xe:
zoomObject(obj.object);
break;
default:
break;
}
}
}
void MacVentureEngine::printTexts() {
for (uint i = 0; i < _textQueue.size(); i++) {
QueuedText text = _textQueue.front();
_textQueue.remove_at(0);
switch (text.id) {
case kTextNumber:
_gui->printText(Common::String(text.asset));
gameChanged();
break;
case kTextNewLine:
_gui->printText(Common::String(""));
gameChanged();
break;
case kTextPlain:
_gui->printText(_world->getText(text.asset, text.source, text.destination));
gameChanged();
break;
default:
break;
}
}
}
void MacVentureEngine::playSounds(bool pause) {
int delay = 0;
while (!_soundQueue.empty()) {
QueuedSound item = _soundQueue.front();
_soundQueue.remove_at(0);
switch (item.id) {
case kSoundPlay:
_soundManager->playSound(item.reference);
break;
case kSoundPlayAndWait:
delay = _soundManager->playSound(item.reference);
break;
case kSoundWait:
// Empty in the original.
break;
default:
break;
}
}
if (pause && delay > 0) {
warning("Sound pausing not yet tested. Pausing for %d", delay);
g_system->delayMillis(delay);
preparedToRun();
}
}
void MacVentureEngine::updateControls() {
selectControl(kNoCommand);
_gui->clearControls();
toggleExits();
resetVars();
}
void MacVentureEngine::resetVars() {
selectControl(kNoCommand);
_currentSelection.clear();
_destObject = 0;
setDeltaPoint(Common::Point(0, 0));
_cmdReady = false;
}
void MacVentureEngine::unselectAll() {
while (!_currentSelection.empty()) {
unselectObject(_currentSelection.front());
}
}
void MacVentureEngine::selectObject(ObjID objID) {
if (!_currentSelection.empty()) {
if (findParentWindow(objID) != findParentWindow(_currentSelection[0])) {
// TODO: Needs further testing, but it doesn't seem necessary.
//unselectAll();
}
}
if (findObjectInArray(objID, _currentSelection) == -1) {
_currentSelection.push_back(objID);
highlightExit(objID);
}
}
void MacVentureEngine::unselectObject(ObjID objID) {
int idxCur = findObjectInArray(objID, _currentSelection);
if (idxCur != -1) {
_currentSelection.remove_at(idxCur);
highlightExit(objID);
}
}
void MacVentureEngine::updateExits() {
_gui->clearExits();
_gui->unselectExits();
Common::Array<ObjID> exits = _world->getChildren(_world->getObjAttr(1, kAttrParentObject), true);
for (uint i = 0; i < exits.size(); i++)
_gui->updateExit(exits[i]);
}
int MacVentureEngine::findObjectInArray(ObjID objID, const Common::Array<ObjID> &list) {
// Find the object in the current selection
bool found = false;
uint i = 0;
while (i < list.size() && !found) {
if (list[i] == objID) {
found = true;
} else {
i++;
}
}
// HACK, should use iterator
return found ? (int)i : -1;
}
uint MacVentureEngine::getPrefixNdx(ObjID obj) {
return _world->getObjAttr(obj, kAttrPrefixes);
}
Common::String MacVentureEngine::getPrefixString(uint flag, ObjID obj) {
uint ndx = getPrefixNdx(obj);
ndx = ((ndx) >> flag) & 3;
return _decodingNamingArticles->getString(ndx);
}
Common::String MacVentureEngine::getNoun(ObjID ndx) {
return _decodingIndirectArticles->getString(ndx);
}
void MacVentureEngine::highlightExit(ObjID objID) {
// TODO: It seems unnecessary since the GUI checks whether an object
// is selected, which includes exits.
warning("STUB: highlightExit");
}
void MacVentureEngine::selectPrimaryObject(ObjID objID) {
if (objID == _destObject) {
return;
}
int idx;
debugC(4, kMVDebugMain, "Select primary object (%d)", objID);
if (_destObject > 0 &&
(idx = findObjectInArray(_destObject, _currentSelection)) != -1) {
unselectAll();
}
_destObject = objID;
if (findObjectInArray(_destObject, _currentSelection) == -1) {
selectObject(_destObject);
}
_cmdReady = true;
}
void MacVentureEngine::focusObjectWindow(ObjID objID) {
if (objID) {
WindowReference win = getObjWindow(objID);
if (win)
_gui->bringToFront(win);
}
}
void MacVentureEngine::openObject(ObjID objID) {
debugC(3, kMVDebugMain, "Open Object[%d] parent[%d] x[%d] y[%d]",
objID,
_world->getObjAttr(objID, kAttrParentObject),
_world->getObjAttr(objID, kAttrPosX),
_world->getObjAttr(objID, kAttrPosY));
if (getObjWindow(objID)) {
return;
}
if (objID == _world->getObjAttr(1, kAttrParentObject)) {
_gui->updateWindowInfo(kMainGameWindow, objID, _world->getChildren(objID, true));
_gui->updateWindow(kMainGameWindow, _world->getObjAttr(objID, kAttrContainerOpen));
updateExits();
_gui->setWindowTitle(kMainGameWindow, _world->getText(objID, objID, objID)); // it ignores source and target in the original
} else { // Open inventory window
Common::Point p(_world->getObjAttr(objID, kAttrPosX), _world->getObjAttr(objID, kAttrPosY));
WindowReference invID = _gui->createInventoryWindow(objID);
Common::String title = _world->getText(objID, objID, objID);
// HACK, trim titletext to fit initial inventory size
while (title.size() > 6) {
title.deleteLastChar();
}
_gui->setWindowTitle(invID, title);
_gui->updateWindowInfo(invID, objID, _world->getChildren(objID, true));
_gui->updateWindow(invID, _world->getObjAttr(objID, kAttrContainerOpen));
}
}
void MacVentureEngine::closeObject(ObjID objID) {
warning("closeObject: not fully implemented");
_gui->tryCloseWindow(getObjWindow(objID));
}
void MacVentureEngine::checkObject(QueuedObject old) {
bool hasChanged = false;
debugC(3, kMVDebugMain, "Check Object[%d] parent[%d] x[%d] y[%d]",
old.object,
old.parent,
old.x,
old.y);
ObjID id = old.object;
if (id == 1) {
if (old.parent != _world->getObjAttr(id, kAttrParentObject)) {
enqueueObject(kSetToPlayerParent, id);
}
if (old.offscreen != !!_world->getObjAttr(id, kAttrInvisible) ||
old.invisible != !!_world->getObjAttr(id, kAttrUnclickable)) {
updateWindow(findParentWindow(id));
}
} else if (old.parent != _world->getObjAttr(id, kAttrParentObject) ||
old.x != _world->getObjAttr(id, kAttrPosX) ||
old.y != _world->getObjAttr(id, kAttrPosY)) {
WindowReference oldWin = getObjWindow(old.parent);
if (oldWin) {
_gui->removeChild(oldWin, id);
hasChanged = true;
}
WindowReference newWin = findParentWindow(id);
if (newWin) {
_gui->addChild(newWin, id);
hasChanged = true;
}
} else if (old.offscreen != !!_world->getObjAttr(id, kAttrInvisible) ||
old.invisible != !!_world->getObjAttr(id, kAttrUnclickable)) {
updateWindow(findParentWindow(id));
}
if (_world->getObjAttr(id, kAttrIsExit)) {
if (hasChanged ||
old.hidden != !!_world->getObjAttr(id, kAttrHiddenExit) ||
old.exitx != _world->getObjAttr(id, kAttrExitX) ||
old.exity != _world->getObjAttr(id, kAttrExitY))
_gui->updateExit(id);
}
WindowReference win = getObjWindow(id);
ObjID cur = id;
ObjID root = _world->getObjAttr(1, kAttrParentObject);
while (cur != root) {
if (cur == 0 || !_world->getObjAttr(cur, kAttrContainerOpen)) {
break;
}
cur = _world->getObjAttr(cur, kAttrParentObject);
}
if (cur == root) {
if (win) {
return;
}
enqueueObject(kOpenWindow, id); //open
} else {
if (!win) {
return;
}
enqueueObject(kCloseWindow, id); //close
}
// Update children
Common::Array<ObjID> children = _world->getChildren(id, true);
for (uint i = 0; i < children.size(); i++) {
enqueueObject(kUpdateObject, children[i]);
}
}
void MacVentureEngine::reflectSwap(ObjID fromID, ObjID toID) {
WindowReference from = getObjWindow(fromID);
WindowReference to = getObjWindow(toID);
WindowReference tmp = to;
debugC(3, kMVDebugMain, "Swap Object[%d] to Object[%d], from win[%d] to win[%d] ",
fromID, toID, from, to);
if (!to) {
tmp = from;
}
if (tmp) {
Common::String newTitle = _world->getText(toID, 0, 0); // Ignores src and targ in the original
_gui->setWindowTitle(tmp, newTitle);
_gui->updateWindowInfo(tmp, toID, _world->getChildren(toID, true));
updateWindow(tmp);
}
}
void MacVentureEngine::toggleExits() {
Common::Array<ObjID> exits = _currentSelection;
while (!exits.empty()) {
ObjID obj = exits.front();
exits.remove_at(0);
highlightExit(obj);
updateWindow(findParentWindow(obj));
}
}
void MacVentureEngine::zoomObject(ObjID objID) {
warning("zoomObject: unimplemented");
}
bool MacVentureEngine::isObjEnqueued(ObjID objID) {
Common::Array<QueuedObject>::const_iterator it;
for (it = _objQueue.begin(); it != _objQueue.end(); it++) {
if ((*it).object == objID) {
return true;
}
}
return false;
}
bool MacVentureEngine::isGameRunning() {
return (_gameState == kGameStateInit || _gameState == kGameStatePlaying);
}
ControlAction MacVenture::MacVentureEngine::referenceToAction(ControlType id) {
switch (id) {
case MacVenture::kControlExitBox:
return kActivateObject;//?? Like this in the original
case MacVenture::kControlExamine:
return kExamine;
case MacVenture::kControlOpen:
return kOpen;
case MacVenture::kControlClose:
return kClose;
case MacVenture::kControlSpeak:
return kSpeak;
case MacVenture::kControlOperate:
return kOperate;
case MacVenture::kControlGo:
return kGo;
case MacVenture::kControlHit:
return kHit;
case MacVenture::kControlConsume:
return kConsume;
default:
return kNoCommand;
}
}
// Data retrieval
bool MacVentureEngine::isPaused() {
return _paused;
}
bool MacVentureEngine::needsClickToContinue() {
return _clickToContinue;
}
Common::String MacVentureEngine::getCommandsPausedString() const {
return Common::String("Click to continue");
}
Common::String MacVentureEngine::getFilePath(FilePathID id) const {
if (id <= 3) { // We don't want a file in the subdirectory
return _filenames->getString(id);
} else { // We want a game file
return _filenames->getString(3) + "/" + _filenames->getString(id);
}
}
bool MacVentureEngine::isOldText() const {
return _oldTextEncoding;
}
const HuffmanLists *MacVentureEngine::getDecodingHuffman() const {
return _textHuffman;
}
uint32 MacVentureEngine::randBetween(uint32 min, uint32 max) {
return _rnd->getRandomNumber(max - min) + min;
}
uint32 MacVentureEngine::getInvolvedObjects() {
// If there is no valid control selected, we return a number too big
// to be useful. There is no control that uses that many objects.
return (_selectedControl ? getGlobalSettings()._cmdArgCnts[_selectedControl - 1] : 3000);
}
Common::Point MacVentureEngine::getObjPosition(ObjID objID) {
return Common::Point(_world->getObjAttr(objID, kAttrPosX), _world->getObjAttr(objID, kAttrPosY));
}
bool MacVentureEngine::isObjVisible(ObjID objID) {
return _world->getObjAttr(objID, kAttrInvisible) == 0;
}
bool MacVentureEngine::isObjClickable(ObjID objID) {
return _world->getObjAttr(objID, kAttrUnclickable) == 0;
}
bool MacVentureEngine::isObjSelected(ObjID objID) {
int idx = findObjectInArray(objID, _currentSelection);
return idx != -1;
}
bool MacVentureEngine::isObjExit(ObjID objID) {
return _world->getObjAttr(objID, kAttrIsExit);
}
bool MacVentureEngine::isHiddenExit(ObjID objID) {
return _world->getObjAttr(objID, kAttrHiddenExit);
}
Common::Point MacVentureEngine::getObjExitPosition(ObjID objID) {
uint x = _world->getObjAttr(objID, kAttrExitX);
uint y = _world->getObjAttr(objID, kAttrExitY);
return Common::Point(x, y);
}
ObjID MacVentureEngine::getParent(ObjID objID) {
return _world->getObjAttr(objID, kAttrParentObject);
}
Common::Rect MacVentureEngine::getObjBounds(ObjID objID) {
Common::Point pos = getObjPosition(objID);
Common::Point measures = _gui->getObjMeasures(objID);
uint w = measures.x;
uint h = measures.y;
return Common::Rect(pos.x, pos.y, pos.x + w, pos.y + h);
}
uint MacVentureEngine::getOverlapPercent(ObjID one, ObjID other) {
// If it's not the same parent, there's 0 overlap
if (_world->getObjAttr(one, kAttrParentObject) !=
_world->getObjAttr(other, kAttrParentObject))
return 0;
Common::Rect oneBounds = getObjBounds(one);
Common::Rect otherBounds = getObjBounds(other);
if (otherBounds.intersects(oneBounds) ||
oneBounds.intersects(otherBounds)) {
uint areaOne = oneBounds.width() * oneBounds.height();
uint areaOther = otherBounds.width() * otherBounds.height();
return (areaOne != 0) ? (areaOther * 100 / areaOne) : 0;
}
return 0;
}
WindowReference MacVentureEngine::getObjWindow(ObjID objID) {
return _gui->getObjWindow(objID);
}
WindowReference MacVentureEngine::findParentWindow(ObjID objID) {
if (objID == 1) {
return kSelfWindow;
}
ObjID parent = _world->getObjAttr(objID, kAttrParentObject);
if (parent == 0) {
return kNoWindow;
}
return getObjWindow(parent);
}
Common::Point MacVentureEngine::getDeltaPoint() {
return _deltaPoint;
}
ObjID MacVentureEngine::getDestObject() {
return _destObject;
}
ControlAction MacVentureEngine::getSelectedControl() {
return _selectedControl;
}
// Data loading
bool MacVentureEngine::loadGlobalSettings() {
Common::MacResIDArray resArray;
if ((resArray = _resourceManager->getResIDArray(MKTAG('G', 'N', 'R', 'L'))).size() == 0)
return false;
Common::SeekableReadStream *res;
res = _resourceManager->getResource(MKTAG('G', 'N', 'R', 'L'), kGlobalSettingsID);
if (res) {
_globalSettings = new GlobalSettings();
_globalSettings->loadSettings(res);
delete res;
return true;
}
return false;
}
bool MacVentureEngine::loadTextHuffman() {
Common::MacResIDArray resArray;
Common::SeekableReadStream *res;
if ((resArray = _resourceManager->getResIDArray(MKTAG('G', 'N', 'R', 'L'))).size() == 0)
return false;
res = _resourceManager->getResource(MKTAG('G', 'N', 'R', 'L'), kTextHuffmanTableID);
if (res) {
uint32 numEntries = res->readUint16BE();
res->readUint16BE(); // Skip
uint32 *masks = new uint32[numEntries];
for (uint i = 0; i < numEntries - 1; i++) {
// For some reason there are one lass mask than entries
masks[i] = res->readUint16BE();
}
uint32 *lengths = new uint32[numEntries];
for (uint i = 0; i < numEntries; i++) {
lengths[i] = res->readByte();
}
uint32 *values = new uint32[numEntries];
for (uint i = 0; i < numEntries; i++) {
values[i] = res->readByte();
}
_textHuffman = new HuffmanLists(numEntries, lengths, masks, values);
debugC(4, kMVDebugMain, "Text is huffman-encoded");
delete res;
delete[] masks;
delete[] lengths;
delete[] values;
return true;
}
return false;
}
// Global Settings
GlobalSettings::GlobalSettings() {
}
GlobalSettings::~GlobalSettings() {
}
void GlobalSettings::loadSettings(Common::SeekableReadStream *dataStream) {
_numObjects = dataStream->readUint16BE();
_numGlobals = dataStream->readUint16BE();
_numCommands = dataStream->readUint16BE();
_numAttributes = dataStream->readUint16BE();
_numGroups = dataStream->readUint16BE();
dataStream->readUint16BE(); // unknown
_invTop = dataStream->readUint16BE();
_invLeft = dataStream->readUint16BE();
_invHeight = dataStream->readUint16BE();
_invWidth = dataStream->readUint16BE();
_invOffsetY = dataStream->readUint16BE();
_invOffsetX = dataStream->readSint16BE();
_defaultFont = dataStream->readUint16BE();
_defaultSize = dataStream->readUint16BE();
uint8 *attrIndices = new uint8[_numAttributes];
dataStream->read(attrIndices, _numAttributes);
_attrIndices = Common::Array<uint8>(attrIndices, _numAttributes);
delete[] attrIndices;
for (int i = 0; i < _numAttributes; i++) {
_attrMasks.push_back(dataStream->readUint16BE());
}
uint8 *attrShifts = new uint8[_numAttributes];
dataStream->read(attrShifts, _numAttributes);
_attrShifts = Common::Array<uint8>(attrShifts, _numAttributes);
delete[] attrShifts;
uint8 *cmdArgCnts = new uint8[_numCommands];
dataStream->read(cmdArgCnts, _numCommands);
_cmdArgCnts = Common::Array<uint8>(cmdArgCnts, _numCommands);
delete[] cmdArgCnts;
uint8 *commands = new uint8[_numCommands];
dataStream->read(commands, _numCommands);
_commands = Common::Array<uint8>(commands, _numCommands);
delete[] commands;
}
} // End of namespace MacVenture