aboutsummaryrefslogtreecommitdiff
path: root/engines/mohawk/myst.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'engines/mohawk/myst.cpp')
-rw-r--r--engines/mohawk/myst.cpp492
1 files changed, 36 insertions, 456 deletions
diff --git a/engines/mohawk/myst.cpp b/engines/mohawk/myst.cpp
index 4ff1df2eb7..d0d3a3c30b 100644
--- a/engines/mohawk/myst.cpp
+++ b/engines/mohawk/myst.cpp
@@ -29,6 +29,7 @@
#include "mohawk/cursors.h"
#include "mohawk/myst.h"
#include "mohawk/myst_areas.h"
+#include "mohawk/myst_card.h"
#include "mohawk/myst_graphics.h"
#include "mohawk/myst_scripts.h"
#include "mohawk/myst_sound.h"
@@ -71,13 +72,8 @@ MohawkEngine_Myst::MohawkEngine_Myst(OSystem *syst, const MohawkGameDescription
_mainCursor = kDefaultMystCursor;
_showResourceRects = false;
_curStack = 0;
- _curCard = 0;
_lastSaveTime = 0;
- _hoverResource = nullptr;
- _activeResource = nullptr;
- _clickedResource = nullptr;
-
_sound = nullptr;
_video = nullptr;
_gfx = nullptr;
@@ -93,7 +89,6 @@ MohawkEngine_Myst::MohawkEngine_Myst(OSystem *syst, const MohawkGameDescription
_mouseMoved = false;
_escapePressed = false;
_waitingOnBlockingOperation = false;
- _runExitScript = true;
_needsPageDrop = false;
_needsShowCredits = false;
@@ -113,9 +108,6 @@ MohawkEngine_Myst::~MohawkEngine_Myst() {
delete _optionsDialog;
delete _prevStack;
delete _rnd;
-
- for (uint32 i = 0; i < _resources.size(); i++)
- delete _resources[i];
}
// Uses cached data objects in preference to disk access
@@ -133,7 +125,6 @@ Common::SeekableReadStream *MohawkEngine_Myst::getResource(uint32 tag, uint16 id
}
error("Could not find a \'%s\' resource with ID %04x", tag2str(tag), id);
- return nullptr;
}
Common::Array<uint16> MohawkEngine_Myst::getResourceIDList(uint32 type) const {
@@ -499,8 +490,16 @@ void MohawkEngine_Myst::doFrame() {
}
if (isInteractive()) {
- updateActiveResource();
- checkCurrentResource();
+ Common::Point mousePos = _system->getEventManager()->getMousePos();
+
+ // Keep a reference to the card so it is not freed if a script switches to another card
+ MystCardPtr card = _card;
+ card->updateActiveResource(mousePos);
+ card->updateResourcesForInput(mousePos, _mouseClicked, _mouseMoved);
+
+ refreshCursor();
+
+ _mouseMoved = false;
}
_system->updateScreen();
@@ -566,6 +565,9 @@ void MohawkEngine_Myst::changeToStack(uint16 stack, uint16 card, uint16 linkSrcS
if (linkSrcSound)
playSoundBlocking(linkSrcSound);
+ _card->leave();
+ _card.reset();
+
// Delete the previous stack and move the current stack to the previous one
// There's probably a better way to do this, but the script classes shouldn't
// take up much memory.
@@ -635,8 +637,6 @@ void MohawkEngine_Myst::changeToStack(uint16 stack, uint16 card, uint16 linkSrcS
if (!_mhk[0]->openFile(mystFiles[_curStack]))
error("Could not open %s", mystFiles[_curStack]);
- _runExitScript = false;
-
// Clear the resource cache and the image cache
_cache.clear();
_gfx->clearCache();
@@ -647,26 +647,6 @@ void MohawkEngine_Myst::changeToStack(uint16 stack, uint16 card, uint16 linkSrcS
playSoundBlocking(linkDstSound);
}
-uint16 MohawkEngine_Myst::getCardBackgroundId() {
- uint16 imageToDraw = 0;
-
- if (_view.conditionalImages.size() == 0)
- imageToDraw = _view.mainImage;
- else {
- for (uint16 i = 0; i < _view.conditionalImages.size(); i++) {
- uint16 varValue = _scriptParser->getVar(_view.conditionalImages[i].var);
- if (varValue < _view.conditionalImages[i].values.size())
- imageToDraw = _view.conditionalImages[i].values[varValue];
- }
- }
-
- return imageToDraw;
-}
-
-void MohawkEngine_Myst::drawCardBackground() {
- _gfx->copyImageToBackBuffer(getCardBackgroundId(), Common::Rect(0, 0, 544, 332));
-}
-
void MohawkEngine_Myst::changeToCard(uint16 card, TransitionType transition) {
debug(2, "changeToCard(%d)", card);
@@ -674,14 +654,6 @@ void MohawkEngine_Myst::changeToCard(uint16 card, TransitionType transition) {
_video->stopVideos();
- // Run exit script from last card (if present)
- if (_runExitScript)
- runExitScript();
-
- _runExitScript = true;
-
- unloadCard();
-
// Clear the resource cache and image cache
_cache.clear();
_gfx->clearCache();
@@ -689,30 +661,13 @@ void MohawkEngine_Myst::changeToCard(uint16 card, TransitionType transition) {
_mouseClicked = false;
_mouseMoved = false;
_escapePressed = false;
- _curCard = card;
-
- // Load a bunch of stuff
- loadCard();
- loadResources();
- loadCursorHints();
- // Handle images
- drawCardBackground();
-
- // Handle sound
- applySoundBlock(_view.soundBlock);
-
- if (_view.flags & kMystZipDestination)
- _gameState->addZipDest(_curStack, card);
-
- // Run the entrance script (if present)
- runInitScript();
-
- // Update the images of each area too
- drawResourceImages();
+ if (_card) {
+ _card->leave();
+ }
- for (uint16 i = 0; i < _resources.size(); i++)
- _resources[i]->handleCardChange();
+ _card = MystCardPtr(new MystCard(this, card));
+ _card->enter();
// The demo resets the cursor at each card change except when in the library
if (getFeatures() & GF_DEMO
@@ -731,310 +686,7 @@ void MohawkEngine_Myst::changeToCard(uint16 card, TransitionType transition) {
// Debug: Show resource rects
if (_showResourceRects)
- drawResourceRects();
-}
-
-void MohawkEngine_Myst::drawResourceRects() {
- for (uint16 i = 0; i < _resources.size(); i++) {
- _resources[i]->getRect().debugPrint(0);
- _resources[i]->drawBoundingRect();
- }
-}
-
-void MohawkEngine_Myst::updateActiveResource() {
- const Common::Point &mouse = _system->getEventManager()->getMousePos();
-
- _activeResource = nullptr;
- for (uint16 i = 0; i < _resources.size(); i++) {
- if (_resources[i]->contains(mouse) && _resources[i]->canBecomeActive()) {
- _activeResource = _resources[i];
- break;
- }
- }
-}
-
-void MohawkEngine_Myst::checkCurrentResource() {
- const Common::Point &mouse = _system->getEventManager()->getMousePos();
-
- // Tell previous resource the mouse is no longer hovering it
- if (_hoverResource && !_hoverResource->contains(mouse)) {
- _hoverResource->handleMouseLeave();
- _hoverResource = nullptr;
- }
-
- for (uint16 i = 0; i < _resources.size(); i++) {
- if (_resources[i]->contains(mouse) && _resources[i]->hasType(kMystAreaHover)
- && _hoverResource != _resources[i]) {
- _hoverResource = static_cast<MystAreaHover *>(_resources[i]);
- _hoverResource->handleMouseEnter();
- }
- }
-
- if (!_mouseClicked && _clickedResource) {
- if (_clickedResource->isEnabled()) {
- _clickedResource->handleMouseUp();
- }
- _clickedResource = nullptr;
- } else if (_mouseMoved && _clickedResource) {
- if (_clickedResource->isEnabled()) {
- _clickedResource->handleMouseDrag();
- }
- } else if (_mouseClicked && !_clickedResource) {
- if (_activeResource && _activeResource->isEnabled()) {
- _clickedResource = _activeResource;
- _clickedResource->handleMouseDown();
- }
- }
-
- _mouseMoved = false;
-
- checkCursorHints();
-}
-
-MystArea *MohawkEngine_Myst::forceUpdateClickedResource() {
- updateActiveResource();
-
- _clickedResource = _activeResource;
-
- return _clickedResource;
-}
-
-void MohawkEngine_Myst::loadCard() {
- debugC(kDebugView, "Loading Card View: %d", _curCard);
-
- Common::SeekableReadStream *viewStream = getResource(ID_VIEW, _curCard);
-
- // Card Flags
- _view.flags = viewStream->readUint16LE();
- debugC(kDebugView, "Flags: 0x%04X", _view.flags);
-
- // The Image Block (Reminiscent of Riven PLST resources)
- uint16 conditionalImageCount = viewStream->readUint16LE();
- debugC(kDebugView, "Conditional Image Count: %d", conditionalImageCount);
- if (conditionalImageCount != 0) {
- for (uint16 i = 0; i < conditionalImageCount; i++) {
- MystCondition conditionalImage;
-
- debugC(kDebugView, "\tImage %d:", i);
- conditionalImage.var = viewStream->readUint16LE();
- debugC(kDebugView, "\t\tVar: %d", conditionalImage.var);
- uint16 numStates = viewStream->readUint16LE();
- debugC(kDebugView, "\t\tNumber of States: %d", numStates);
- for (uint16 j = 0; j < numStates; j++) {
- conditionalImage.values.push_back(viewStream->readUint16LE());
- debugC(kDebugView, "\t\tState %d -> Value %d", j, conditionalImage.values[j]);
- }
-
- _view.conditionalImages.push_back(conditionalImage);
- }
- _view.mainImage = 0;
- } else {
- _view.mainImage = viewStream->readUint16LE();
- debugC(kDebugView, "Main Image: %d", _view.mainImage);
- }
-
- // The Sound Block (Reminiscent of Riven SLST resources)
- _view.soundBlock = readSoundBlock(viewStream);
-
- // Resources that scripts can call upon
- uint16 scriptResCount = viewStream->readUint16LE();
- debugC(kDebugView, "Script Resource Count: %d", scriptResCount);
- for (uint16 i = 0; i < scriptResCount; i++) {
- MystView::ScriptResource scriptResource;
-
- debugC(kDebugView, "\tResource %d:", i);
- scriptResource.type = (MystView::ScriptResourceType) viewStream->readUint16LE();
- debugC(kDebugView, "\t\t Type: %d", scriptResource.type);
-
- switch (scriptResource.type) {
- case MystView::kResourceImage:
- debugC(kDebugView, "\t\t\t\t= Image");
- break;
- case MystView::kResourceSound:
- debugC(kDebugView, "\t\t\t\t= Sound");
- break;
- case MystView::kResourceSwitch:
- debugC(kDebugView, "\t\t\t\t= Resource Switch");
- break;
- case MystView::kResourceImageNoCache:
- debugC(kDebugView, "\t\t\t\t= Image - Caching disabled");
- break;
- case MystView::kResourceSoundNoCache:
- debugC(kDebugView, "\t\t\t\t= Sound - Caching disabled");
- break;
- default:
- debugC(kDebugView, "\t\t\t\t= Unknown");
- warning("Unknown script resource type '%d' in card '%d'", scriptResource.type, _curCard);
- break;
- }
-
- if (scriptResource.type == MystView::kResourceSwitch) {
- scriptResource.switchVar = viewStream->readUint16LE();
- debugC(kDebugView, "\t\t Var: %d", scriptResource.switchVar);
- uint16 count = viewStream->readUint16LE();
- debugC(kDebugView, "\t\t Resource List Count: %d", count);
- scriptResource.switchResourceType = (MystView::ScriptResourceType) viewStream->readUint16LE();
- debugC(kDebugView, "\t\t u0: %d", scriptResource.switchResourceType);
-
- for (uint16 j = 0; j < count; j++) {
- scriptResource.switchResourceIds.push_back(viewStream->readSint16LE());
- debugC(kDebugView, "\t\t Resource List %d: %d", j, scriptResource.switchResourceIds[j]);
- }
- } else {
- scriptResource.id = viewStream->readUint16LE();
- debugC(kDebugView, "\t\t Id: %d", scriptResource.id);
- }
-
- _view.scriptResources.push_back(scriptResource);
- }
-
- // Identifiers for other resources. 0 if non existent. There is always an RLST.
- _view.rlst = viewStream->readUint16LE();
- if (!_view.rlst)
- error("RLST Index missing");
-
- _view.hint = viewStream->readUint16LE();
- _view.init = viewStream->readUint16LE();
- _view.exit = viewStream->readUint16LE();
-
- delete viewStream;
-
- // Precache Card Resources
- uint32 cacheImageType;
- if (getFeatures() & GF_ME)
- cacheImageType = ID_PICT;
- else
- cacheImageType = ID_WDIB;
-
- // Precache Image Block data
- if (_view.conditionalImages.size() != 0) {
- for (uint16 i = 0; i < _view.conditionalImages.size(); i++) {
- uint16 value = _scriptParser->getVar(_view.conditionalImages[i].var);
- cachePreload(cacheImageType, _view.conditionalImages[i].values[value]);
- }
- } else {
- cachePreload(cacheImageType, _view.mainImage);
- }
-
- // Precache Sound Block data
- if (_view.soundBlock.sound > 0)
- cachePreload(ID_MSND, _view.soundBlock.sound);
- else if (_view.soundBlock.sound == kMystSoundActionConditional) {
- uint16 value = _scriptParser->getVar(_view.soundBlock.soundVar);
- if (_view.soundBlock.soundList[value].action > 0) {
- cachePreload(ID_MSND, _view.soundBlock.soundList[value].action);
- }
- }
-
- // Precache Script Resources
- for (uint16 i = 0; i < _view.scriptResources.size(); i++) {
- MystView::ScriptResourceType type;
- int16 id;
- if (_view.scriptResources[i].type == MystView::kResourceSwitch) {
- type = _view.scriptResources[i].switchResourceType;
- uint16 value = _scriptParser->getVar(_view.scriptResources[i].switchVar);
- id = _view.scriptResources[i].switchResourceIds[value];
- } else {
- type = _view.scriptResources[i].type;
- id = _view.scriptResources[i].id;
- }
-
- if (id < 0) continue;
-
- switch (type) {
- case MystView::kResourceImage:
- cachePreload(cacheImageType, id);
- break;
- case MystView::kResourceSound:
- cachePreload(ID_MSND, id);
- break;
- default:
- // The other resource types should not be cached
- break;
- }
- }
-}
-
-void MohawkEngine_Myst::unloadCard() {
- _view.conditionalImages.clear();
- _view.soundBlock.soundList.clear();
- _view.scriptResources.clear();
- _hoverResource = nullptr;
- _activeResource = nullptr;
- _clickedResource = nullptr;
-}
-
-void MohawkEngine_Myst::runInitScript() {
- if (!_view.init) {
- debugC(kDebugINIT, "No INIT Present");
- return;
- }
-
- debugC(kDebugINIT, "Running INIT script");
-
- Common::SeekableReadStream *initStream = getResource(ID_INIT, _view.init);
- MystScript script = _scriptParser->readScript(initStream, kMystScriptInit);
- delete initStream;
-
- _scriptParser->runScript(script);
-}
-
-void MohawkEngine_Myst::runExitScript() {
- if (!_view.exit) {
- debugC(kDebugEXIT, "No EXIT Present");
- return;
- }
-
- debugC(kDebugEXIT, "Running EXIT script");
-
- Common::SeekableReadStream *exitStream = getResource(ID_EXIT, _view.exit);
- MystScript script = _scriptParser->readScript(exitStream, kMystScriptExit);
- delete exitStream;
-
- _scriptParser->runScript(script);
-}
-
-void MohawkEngine_Myst::loadCursorHints() {
- _cursorHints.clear();
-
- if (!_view.hint) {
- debugC(kDebugHint, "No HINT Present");
- return;
- }
-
- debugC(kDebugHint, "Loading Cursor Hints:");
-
- Common::SeekableReadStream *hintStream = getResource(ID_HINT, _curCard);
- uint16 cursorHintCount = hintStream->readUint16LE();
- debugC(kDebugHint, "Cursor Hint Count: %d", cursorHintCount);
-
- for (uint16 i = 0; i < cursorHintCount; i++) {
- MystCursorHint hint;
-
- debugC(kDebugHint, "Cursor Hint %d:", i);
- hint.id = hintStream->readUint16LE();
- debugC(kDebugHint, "\tId: %d", hint.id);
- hint.cursor = hintStream->readSint16LE();
- debugC(kDebugHint, "\tCursor: %d", hint.cursor);
-
- if (hint.cursor == -1) {
- debugC(kDebugHint, "\tConditional Cursor Hints:");
- hint.variableHint.var = hintStream->readUint16LE();
- debugC(kDebugHint, "\tVar: %d", hint.variableHint.var);
- uint16 numStates = hintStream->readUint16LE();
- debugC(kDebugHint, "\tNumber of States: %d", numStates);
- for (uint16 j = 0; j < numStates; j++) {
- hint.variableHint.values.push_back(hintStream->readUint16LE());
- debugC(kDebugHint, "\t\t State %d: Cursor %d", j, hint.variableHint.values[j]);
- }
- } else {
- hint.variableHint.var = 0;
- }
-
- _cursorHints.push_back(hint);
- }
-
- delete hintStream;
+ _card->drawResourceRects();
}
void MohawkEngine_Myst::setMainCursor(uint16 cursor) {
@@ -1042,70 +694,22 @@ void MohawkEngine_Myst::setMainCursor(uint16 cursor) {
_cursor->setCursor(_currentCursor);
}
-void MohawkEngine_Myst::checkCursorHints() {
- if (!_view.hint) {
- // Default to the main cursor when no hints are present
- if (_currentCursor != _mainCursor) {
- _currentCursor = _mainCursor;
- _cursor->setCursor(_currentCursor);
- }
- return;
+void MohawkEngine_Myst::refreshCursor() {
+ int16 cursor = _card->getActiveResourceCursor();
+ if (cursor == -1) {
+ cursor = _mainCursor;
}
- // Check all the cursor hints to see if we're in a hotspot that contains a hint.
- for (uint16 i = 0; i < _cursorHints.size(); i++)
- if (_activeResource && _resources[_cursorHints[i].id] == _activeResource && _activeResource->isEnabled()) {
- if (_cursorHints[i].cursor == -1) {
- uint16 var_value = _scriptParser->getVar(_cursorHints[i].variableHint.var);
-
- if (var_value >= _cursorHints[i].variableHint.values.size())
- warning("Variable %d Out of Range in variable HINT Resource %d", _cursorHints[i].variableHint.var, i);
- else {
- _currentCursor = _cursorHints[i].variableHint.values[var_value];
- if (_currentCursor == 0)
- _currentCursor = _mainCursor;
- _cursor->setCursor(_currentCursor);
- }
- } else if (_currentCursor != _cursorHints[i].cursor) {
- if (_cursorHints[i].cursor == 0)
- _currentCursor = _mainCursor;
- else
- _currentCursor = _cursorHints[i].cursor;
-
- _cursor->setCursor(_currentCursor);
- }
- return;
- }
-
- if (_currentCursor != _mainCursor) {
- _currentCursor = _mainCursor;
- _cursor->setCursor(_currentCursor);
+ if (cursor != _currentCursor) {
+ _currentCursor = cursor;
+ _cursor->setCursor(cursor);
}
}
-void MohawkEngine_Myst::setResourceEnabled(uint16 resourceId, bool enable) {
- if (resourceId < _resources.size()) {
- _resources[resourceId]->setEnabled(enable);
- } else
- warning("Attempt to change unknown resource enable state");
-}
-
-void MohawkEngine_Myst::drawResourceImages() {
- for (uint16 i = 0; i < _resources.size(); i++)
- if (_resources[i]->isDrawSubimages())
- _resources[i]->drawDataToScreen();
-}
-
void MohawkEngine_Myst::redrawResource(MystAreaImageSwitch *resource, bool update) {
resource->drawConditionalDataToScreen(_scriptParser->getVar(resource->getImageSwitchVar()), update);
}
-void MohawkEngine_Myst::redrawArea(uint16 var, bool update) {
- for (uint16 i = 0; i < _resources.size(); i++)
- if (_resources[i]->hasType(kMystAreaImageSwitch) && _resources[i]->getImageSwitchVar() == var)
- redrawResource(static_cast<MystAreaImageSwitch *>(_resources[i]), update);
-}
-
MystArea *MohawkEngine_Myst::loadResource(Common::SeekableReadStream *rlstStream, MystArea *parent) {
MystArea *resource = nullptr;
ResourceType type = static_cast<ResourceType>(rlstStream->readUint16LE());
@@ -1146,29 +750,6 @@ MystArea *MohawkEngine_Myst::loadResource(Common::SeekableReadStream *rlstStream
return resource;
}
-void MohawkEngine_Myst::loadResources() {
- for (uint32 i = 0; i < _resources.size(); i++)
- delete _resources[i];
-
- _resources.clear();
-
- if (!_view.rlst) {
- debugC(kDebugResource, "No RLST present");
- return;
- }
-
- Common::SeekableReadStream *rlstStream = getResource(ID_RLST, _view.rlst);
- uint16 resourceCount = rlstStream->readUint16LE();
- debugC(kDebugResource, "RLST Resource Count: %d", resourceCount);
-
- for (uint16 i = 0; i < resourceCount; i++) {
- debugC(kDebugResource, "Resource #%d:", i);
- _resources.push_back(loadResource(rlstStream, nullptr));
- }
-
- delete rlstStream;
-}
-
Common::Error MohawkEngine_Myst::loadGameState(int slot) {
if (_gameState->load(slot))
return Common::kNoError;
@@ -1208,8 +789,7 @@ bool MohawkEngine_Myst::canLoadGameStateCurrently() {
return false;
}
- if (_clickedResource) {
- // Can't save while dragging resources
+ if (_card->isDraggingResource()) {
return false;
}
@@ -1255,28 +835,28 @@ void MohawkEngine_Myst::dropPage() {
// Redraw page area
if (whitePage && _gameState->_globals.currentAge == kMystLibrary) {
_scriptParser->toggleVar(41);
- redrawArea(41);
+ _card->redrawArea(41);
} else if (bluePage) {
if (page == kBlueFirePlacePage) {
if (_gameState->_globals.currentAge == kMystLibrary)
- redrawArea(24);
+ _card->redrawArea(24);
} else {
- redrawArea(103);
+ _card->redrawArea(103);
}
} else if (redPage) {
if (page == kRedFirePlacePage) {
if (_gameState->_globals.currentAge == kMystLibrary)
- redrawArea(25);
+ _card->redrawArea(25);
} else if (page == kRedStoneshipPage) {
if (_gameState->_globals.currentAge == kStoneship)
- redrawArea(35);
+ _card->redrawArea(35);
} else {
- redrawArea(102);
+ _card->redrawArea(102);
}
}
setMainCursor(kDefaultMystCursor);
- checkCursorHints();
+ refreshCursor();
}
MystSoundBlock MohawkEngine_Myst::readSoundBlock(Common::ReadStream *stream) const {
@@ -1317,7 +897,7 @@ MystSoundBlock MohawkEngine_Myst::readSoundBlock(Common::ReadStream *stream) con
soundBlock.soundList.push_back(sound);
}
} else {
- error("Unknown sound control value '%d' in card '%d'", soundBlock.sound, _curCard);
+ error("Unknown sound control value '%d' in card '%d'", soundBlock.sound, _card->getId());
}
return soundBlock;