mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-05 17:20:30 +00:00
1417 lines
40 KiB
C++
1417 lines
40 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 "mohawk/riven_card.h"
|
|
|
|
#include "mohawk/cursors.h"
|
|
#include "mohawk/riven_graphics.h"
|
|
#include "mohawk/riven_stack.h"
|
|
#include "mohawk/riven_stacks/aspit.h"
|
|
#include "mohawk/riven_video.h"
|
|
|
|
#include "mohawk/resource.h"
|
|
#include "mohawk/riven.h"
|
|
|
|
#include "common/memstream.h"
|
|
|
|
namespace Mohawk {
|
|
|
|
RivenCard::RivenCard(MohawkEngine_Riven *vm, uint16 id) :
|
|
_vm(vm),
|
|
_id(id),
|
|
_hoveredHotspot(nullptr),
|
|
_pressedHotspot(nullptr) {
|
|
loadCardResource(id);
|
|
loadHotspots(id);
|
|
loadCardPictureList(id);
|
|
loadCardSoundList(id);
|
|
loadCardMovieList(id);
|
|
loadCardHotspotEnableList(id);
|
|
loadCardWaterEffectList(id);
|
|
applyPatches(id);
|
|
}
|
|
|
|
RivenCard::~RivenCard() {
|
|
for (uint i = 0; i < _hotspots.size(); i++) {
|
|
delete _hotspots[i];
|
|
}
|
|
|
|
_vm->_gfx->clearWaterEffect();
|
|
_vm->_gfx->clearFliesEffect();
|
|
_vm->_video->closeVideos();
|
|
}
|
|
|
|
void RivenCard::loadCardResource(uint16 id) {
|
|
Common::SeekableReadStream *inStream = _vm->getResource(ID_CARD, id);
|
|
|
|
_name = inStream->readSint16BE();
|
|
_zipModePlace = inStream->readUint16BE();
|
|
_scripts = _vm->_scriptMan->readScripts(inStream);
|
|
|
|
delete inStream;
|
|
}
|
|
|
|
void RivenCard::applyPatches(uint16 id) {
|
|
uint32 globalId = _vm->getStack()->getCardGlobalId(id);
|
|
|
|
if (globalId == 0x2A3BC) {
|
|
applyPropertiesPatch8EB7(globalId, "jladder", 3);
|
|
} else if (globalId == 0x8EB7) {
|
|
applyPropertiesPatch8EB7(globalId, "jgate", 3);
|
|
}
|
|
applyPropertiesPatch2E76(globalId);
|
|
|
|
// Apply script patches
|
|
for (uint i = 0; i < _scripts.size(); i++) {
|
|
_scripts[i].script->applyCardPatches(_vm, globalId, _scripts[i].type, 0xFFFF);
|
|
}
|
|
|
|
applyPropertiesPatch22118(globalId);
|
|
applyPropertiesPatchE2E(globalId);
|
|
applyPropertiesPatch1518D(globalId);
|
|
applyPropertiesPatch2B414(globalId);
|
|
}
|
|
|
|
void RivenCard::applyPropertiesPatch8EB7(uint32 globalId, const Common::String &var, uint16 hotspotId) {
|
|
// On Jungle Island on the back side of the "beetle" gate, the forward hotspot
|
|
// is always enabled, preventing keyboard navigation from automatically opening
|
|
// the gate.
|
|
// We patch the card so that the forward opcode is enabled only when the gate is open.
|
|
//
|
|
// New hotspot enable entries:
|
|
// == Hotspot enable 5 ==
|
|
// hotspotId: 3
|
|
// enabled: 1
|
|
//
|
|
// == Hotspot enable 6 ==
|
|
// hotspotId: 3
|
|
// enabled: 0
|
|
//
|
|
// Additional load script fragment:
|
|
// switch (jgate) {
|
|
// case 0:
|
|
// activateBLST(6);
|
|
// break;
|
|
// case 1:
|
|
// activateBLST(5);
|
|
// break;
|
|
|
|
HotspotEnableRecord forwardEnabled;
|
|
forwardEnabled.index = _hotspotEnableList.back().index + 1;
|
|
forwardEnabled.hotspotId = hotspotId;
|
|
forwardEnabled.enabled = 1;
|
|
_hotspotEnableList.push_back(forwardEnabled);
|
|
|
|
HotspotEnableRecord forwardDisabled;
|
|
forwardDisabled.index = _hotspotEnableList.back().index + 1;
|
|
forwardDisabled.hotspotId = hotspotId;
|
|
forwardDisabled.enabled = 0;
|
|
_hotspotEnableList.push_back(forwardDisabled);
|
|
|
|
uint16 jGateVariable = _vm->getStack()->getIdFromName(kVariableNames, var);
|
|
uint16 patchData[] = {
|
|
1, // Command count in script
|
|
kRivenCommandSwitch,
|
|
2, // Unused
|
|
jGateVariable,
|
|
2, // Branches count
|
|
|
|
0, // jgate == 0 branch (gate closed)
|
|
1, // Command count in sub-script
|
|
kRivenCommandActivateBLST,
|
|
1, // Argument count
|
|
forwardDisabled.index,
|
|
|
|
1, // jgate == 1 branch (gate open)
|
|
1, // Command count in sub-script
|
|
kRivenCommandActivateBLST,
|
|
1, // Argument count
|
|
forwardEnabled.index
|
|
};
|
|
|
|
RivenScriptPtr patchScript = _vm->_scriptMan->readScriptFromData(patchData, ARRAYSIZE(patchData));
|
|
|
|
// Append the patch to the existing script
|
|
RivenScriptPtr loadScript = getScript(kCardLoadScript);
|
|
loadScript += patchScript;
|
|
|
|
debugC(kRivenDebugPatches, "Applied fix always enabled forward hotspot in card %x", globalId);
|
|
}
|
|
|
|
void RivenCard::applyPropertiesPatch2E76(uint32 globalId) {
|
|
// In Gehn's office, after having encountered him once before and coming back
|
|
// with the trap book, the draw update script of card 1 tries to switch to
|
|
// card 2 while still loading card 1. Switching cards is not allowed during
|
|
// draw update scripts, resulting in an use after free crash.
|
|
//
|
|
// Here we backport the fix that has been made in the DVD version to the CD version.
|
|
//
|
|
// Script before patch:
|
|
// == Script 1 ==
|
|
// type: CardUpdate
|
|
// switch (agehn) {
|
|
// case 1:
|
|
// switch (atrapbook) {
|
|
// case 1:
|
|
// obutton = 1;
|
|
// transition(16);
|
|
// switchCard(2);
|
|
// break;
|
|
// }
|
|
// break;
|
|
// case 2:
|
|
// activatePLST(5);
|
|
// break;
|
|
// case 3:
|
|
// activatePLST(5);
|
|
// break;
|
|
// }
|
|
//
|
|
//
|
|
// Script after patch:
|
|
// == Script 1 ==
|
|
// type: CardUpdate
|
|
// switch (agehn) {
|
|
// case 1:
|
|
// switch (atrapbook) {
|
|
// case 1:
|
|
// obutton = 1;
|
|
// activatePLST(6);
|
|
// break;
|
|
// }
|
|
// break;
|
|
// case 2:
|
|
// activatePLST(5);
|
|
// break;
|
|
// case 3:
|
|
// activatePLST(5);
|
|
// break;
|
|
// }
|
|
//
|
|
// == Script 2 ==
|
|
// type: CardEnter
|
|
// switch (agehn) {
|
|
// case 1:
|
|
// switch (atrapbook) {
|
|
// case 1:
|
|
// transition(16);
|
|
// switchCard(2);
|
|
// break;
|
|
// }
|
|
// break;
|
|
// }
|
|
if (globalId == 0x2E76 && !_vm->isGameVariant(GF_DVD)) {
|
|
uint16 aGehnVariable = _vm->getStack()->getIdFromName(kVariableNames, "agehn");
|
|
uint16 aTrapBookVariable = _vm->getStack()->getIdFromName(kVariableNames, "atrapbook");
|
|
uint16 patchData[] = {
|
|
1, // Command count in script
|
|
kRivenCommandSwitch,
|
|
2, // Unused
|
|
aGehnVariable,
|
|
1, // Branches count
|
|
|
|
1, // agehn == 1 branch
|
|
1, // Command count in sub-script
|
|
kRivenCommandSwitch,
|
|
2, // Unused
|
|
aTrapBookVariable,
|
|
1, // Branches count
|
|
|
|
1, // atrapbook == 1 branch
|
|
2, // Command count in sub-script
|
|
kRivenCommandTransition,
|
|
1, // Argument count
|
|
kRivenTransitionBlend,
|
|
kRivenCommandChangeCard,
|
|
1, // Argument count
|
|
2 // Card id
|
|
};
|
|
|
|
// Add the new script to the list
|
|
RivenTypedScript patchScript;
|
|
patchScript.type = kCardEnterScript;
|
|
patchScript.script = _vm->_scriptMan->readScriptFromData(patchData, ARRAYSIZE(patchData));
|
|
_scripts.push_back(patchScript);
|
|
|
|
// Add a black picture to the card's list to be able to use it in the second part of the patch
|
|
Picture blackPicture;
|
|
blackPicture.index = 6;
|
|
blackPicture.id = 117;
|
|
blackPicture.rect = Common::Rect(608, 392);
|
|
_pictureList.push_back(blackPicture);
|
|
|
|
debugC(kRivenDebugPatches, "Applied invalid card change during screen update (1/2) to card %x", globalId);
|
|
// The second part of this patch is in the script patches
|
|
}
|
|
}
|
|
|
|
void RivenCard::applyPropertiesPatch22118(uint32 globalId) {
|
|
// On Temple Island, near the steam valve closest to the bridge to Boiler island,
|
|
// the background sound on the view offering a view to the bridge does
|
|
// not properly reflect the valve's position.
|
|
//
|
|
// The sound is always that of steam going through the pipe when the bridge is
|
|
// down. When the valve points up, the sound should be that of steam escaping
|
|
// through the top of the pipe.
|
|
//
|
|
// Script before patch:
|
|
// == Script 0 ==
|
|
// type: CardLoad
|
|
// switch (bbigbridge) {
|
|
// case 0:
|
|
// switch (tbookvalve) {
|
|
// case 0:
|
|
// activatePLST(2);
|
|
// activateSLST(1);
|
|
// break;
|
|
// }
|
|
// break;
|
|
// }
|
|
// switch (bbigbridge) {
|
|
// case 0:
|
|
// switch (tbookvalve) {
|
|
// case 1:
|
|
// activatePLST(2);
|
|
// activateSLST(2);
|
|
// break;
|
|
// }
|
|
// break;
|
|
// }
|
|
// switch (bbigbridge) {
|
|
// case 1:
|
|
// switch (tbookvalve) {
|
|
// case 0:
|
|
// activatePLST(1);
|
|
// activateSLST(2);
|
|
// break;
|
|
// }
|
|
// break;
|
|
// }
|
|
// switch (bbigbridge) {
|
|
// case 1:
|
|
// switch (tbookvalve) {
|
|
// case 1:
|
|
// activatePLST(1);
|
|
// activateSLST(2);
|
|
// break;
|
|
// }
|
|
// break;
|
|
// }
|
|
//
|
|
//
|
|
// Script after patch:
|
|
// == Script 0 ==
|
|
// type: CardLoad
|
|
// switch (bbigbridge) {
|
|
// case 0:
|
|
// switch (tbookvalve) {
|
|
// case 0:
|
|
// activatePLST(2);
|
|
// break;
|
|
// }
|
|
// break;
|
|
// }
|
|
// switch (bbigbridge) {
|
|
// case 0:
|
|
// switch (tbookvalve) {
|
|
// case 1:
|
|
// activatePLST(2);
|
|
// break;
|
|
// }
|
|
// break;
|
|
// }
|
|
// switch (bbigbridge) {
|
|
// case 1:
|
|
// switch (tbookvalve) {
|
|
// case 0:
|
|
// activatePLST(1);
|
|
// break;
|
|
// }
|
|
// break;
|
|
// }
|
|
// switch (bbigbridge) {
|
|
// case 1:
|
|
// switch (tbookvalve) {
|
|
// case 1:
|
|
// activatePLST(1);
|
|
// break;
|
|
// }
|
|
// break;
|
|
// }
|
|
// switch (tbookvalve) {
|
|
// case 0:
|
|
// activateSLST(1);
|
|
// break;
|
|
// case 1:
|
|
// activateSLST(2);
|
|
// break;
|
|
// }
|
|
if (globalId == 0x22118) {
|
|
uint16 tBookValveVariable = _vm->getStack()->getIdFromName(kVariableNames, "tbookvalve");
|
|
uint16 patchData[] = {
|
|
1, // Command count in script
|
|
kRivenCommandSwitch,
|
|
2, // Unused
|
|
tBookValveVariable,
|
|
2, // Branches count
|
|
|
|
0, // tbookvalve == 0 branch (steam escaping at the top of the pipe)
|
|
1, // Command count in sub-script
|
|
kRivenCommandActivateSLST,
|
|
1, // Argument count
|
|
1, // Steam leaking sound id
|
|
|
|
1, // tbookvalve == 1 branch (steam going to the left pipe)
|
|
1, // Command count in sub-script
|
|
kRivenCommandActivateSLST,
|
|
1, // Argument count
|
|
2, // Steam in pipe sound id
|
|
};
|
|
|
|
RivenScriptPtr patchScript = _vm->_scriptMan->readScriptFromData(patchData, ARRAYSIZE(patchData));
|
|
|
|
// Append the patch to the existing script
|
|
RivenScriptPtr loadScript = getScript(kCardLoadScript);
|
|
loadScript += patchScript;
|
|
|
|
debugC(kRivenDebugPatches, "Applied incorrect steam sounds (2/2) to card %x", globalId);
|
|
}
|
|
}
|
|
|
|
void RivenCard::applyPropertiesPatchE2E(uint32 globalId) {
|
|
if (!_vm->isGameVariant(GF_25TH))
|
|
return;
|
|
|
|
// The main menu in the Myst 25th anniversary version is patched to include new items:
|
|
// - Save game
|
|
if (globalId == 0xE2E) {
|
|
moveHotspot( 22, Common::Rect(470, 175, 602, 190)); // Setup
|
|
moveHotspot( 16, Common::Rect(470, 201, 602, 216)); // New game
|
|
addMenuHotspot(23, Common::Rect(470, 227, 602, 242), 3, RivenStacks::ASpit::kExternalRestoreGame, "xarestoregame");
|
|
addMenuHotspot(24, Common::Rect(470, 256, 602, 271), 4, RivenStacks::ASpit::kExternalSaveGame, "xaSaveGame");
|
|
addMenuHotspot(25, Common::Rect(470, 283, 602, 300), 5, RivenStacks::ASpit::kExternalResume, "xaResumeGame");
|
|
addMenuHotspot(26, Common::Rect(470, 309, 602, 326), 6, RivenStacks::ASpit::kExternalOptions, "xaOptions");
|
|
addMenuHotspot(27, Common::Rect(470, 335, 602, 352), 7, RivenStacks::ASpit::kExternalQuit, "xademoquit");
|
|
_vm->getStack()->registerName(kExternalCommandNames, RivenStacks::ASpit::kExternalNewGame, "xaNewGame");
|
|
}
|
|
}
|
|
|
|
void RivenCard::applyPropertiesPatch1518D(uint32 globalId) {
|
|
// Inside Jungle Island's dome, when looking at the open book,
|
|
// stepping back from the stand and then looking at the book
|
|
// again, the book closing animation would play again.
|
|
//
|
|
// Comparing the scripts for the Jungle dome and the other domes
|
|
// shows a small portion of script is missing.
|
|
// The following patch adds it back so the jungle dome script
|
|
// matches the other domes.
|
|
//
|
|
// Added script part:
|
|
// == Script 2 ==
|
|
// [...]
|
|
// type: CardEnter
|
|
// switch (jbook) {
|
|
// case 2:
|
|
// playMovieBlocking(1);
|
|
// jbook = 0;
|
|
// refreshCard();
|
|
// break;
|
|
// }
|
|
if (globalId == 0x1518D) {
|
|
uint16 jBookVariable = _vm->getStack()->getIdFromName(kVariableNames, "jbook");
|
|
uint16 patchData[] = {
|
|
1, // Command count in script
|
|
kRivenCommandSwitch,
|
|
2, // Unused
|
|
jBookVariable,
|
|
1, // Branches count
|
|
|
|
2, // jbook == 2 branch
|
|
3, // Command count in sub-script
|
|
|
|
kRivenCommandPlayMovieBlocking,
|
|
1, // Argument count
|
|
1, // Video id
|
|
|
|
kRivenCommandSetVariable,
|
|
2, // Argument count
|
|
jBookVariable,
|
|
0, // Variable value
|
|
|
|
kRivenCommandRefreshCard,
|
|
0 // Argument count
|
|
};
|
|
|
|
RivenScriptPtr patchScript = _vm->_scriptMan->readScriptFromData(patchData, ARRAYSIZE(patchData));
|
|
|
|
// Append the patch to the existing script
|
|
RivenScriptPtr loadScript = getScript(kCardEnterScript);
|
|
loadScript += patchScript;
|
|
|
|
debugC(kRivenDebugPatches, "Applied jungle book close loop to card %x", globalId);
|
|
}
|
|
}
|
|
|
|
void RivenCard::applyPropertiesPatch2B414(uint32 globalId) {
|
|
//
|
|
// When seated in the Jungle Island's gallows control throne,
|
|
// the right lever opens or closes the gallows' floor.
|
|
// The issue is that it is possible to click on the position
|
|
// where the lever would be when the gallow's floor in the
|
|
// opposite state as it currently is. And thus to trigger the
|
|
// corresponding sequence. That is to say for example closing
|
|
// the floor when it is already closed.
|
|
//
|
|
// We simply add the missing script instructions to make
|
|
// the open and close hotspots mutually exclusive.
|
|
//
|
|
// Added script part:
|
|
// == Script 0 ==
|
|
// type: CardLoad
|
|
// [...]
|
|
// switch (jgallows) {
|
|
// case 0:
|
|
// activateBLST(1);
|
|
// activateBLST(4);
|
|
// break;
|
|
// case 1:
|
|
// activateBLST(2);
|
|
// activateBLST(3);
|
|
// break;
|
|
// }
|
|
if (globalId == 0x2B414) {
|
|
HotspotEnableRecord openGallowsEnabled;
|
|
openGallowsEnabled.index = 1;
|
|
openGallowsEnabled.hotspotId = 8;
|
|
openGallowsEnabled.enabled = 1;
|
|
_hotspotEnableList.push_back(openGallowsEnabled);
|
|
|
|
HotspotEnableRecord openGallowsDisabled;
|
|
openGallowsDisabled.index = 2;
|
|
openGallowsDisabled.hotspotId = 8;
|
|
openGallowsDisabled.enabled = 0;
|
|
_hotspotEnableList.push_back(openGallowsDisabled);
|
|
|
|
HotspotEnableRecord closeGallowsEnabled;
|
|
closeGallowsEnabled.index = 3;
|
|
closeGallowsEnabled.hotspotId = 9;
|
|
closeGallowsEnabled.enabled = 1;
|
|
_hotspotEnableList.push_back(closeGallowsEnabled);
|
|
|
|
HotspotEnableRecord closeGallowsDisabled;
|
|
closeGallowsDisabled.index = 4;
|
|
closeGallowsDisabled.hotspotId = 9;
|
|
closeGallowsDisabled.enabled = 0;
|
|
_hotspotEnableList.push_back(closeGallowsDisabled);
|
|
|
|
uint16 jGallowsVariable = _vm->getStack()->getIdFromName(kVariableNames, "jgallows");
|
|
uint16 patchData[] = {
|
|
1, // Command count in script
|
|
kRivenCommandSwitch,
|
|
2, // Unused
|
|
jGallowsVariable,
|
|
2, // Branches count
|
|
|
|
0, // jgallows == 0 branch
|
|
2, // Command count in sub-script
|
|
|
|
kRivenCommandActivateBLST,
|
|
1, // Argument count
|
|
openGallowsEnabled.index,
|
|
|
|
kRivenCommandActivateBLST,
|
|
1, // Argument count
|
|
closeGallowsDisabled.index,
|
|
|
|
1, // jgallows == 1 branch
|
|
2, // Command count in sub-script
|
|
|
|
kRivenCommandActivateBLST,
|
|
1, // Argument count
|
|
openGallowsDisabled.index,
|
|
|
|
kRivenCommandActivateBLST,
|
|
1, // Argument count
|
|
closeGallowsEnabled.index,
|
|
|
|
};
|
|
|
|
RivenScriptPtr patchScript = _vm->_scriptMan->readScriptFromData(patchData, ARRAYSIZE(patchData));
|
|
|
|
// Append the patch to the existing script
|
|
RivenScriptPtr loadScript = getScript(kCardLoadScript);
|
|
loadScript += patchScript;
|
|
|
|
debugC(kRivenDebugPatches, "Applied missing jgallows hotspot enable / disable to card %x", globalId);
|
|
}
|
|
}
|
|
|
|
void RivenCard::moveHotspot(uint16 blstId, const Common::Rect &position) {
|
|
RivenHotspot *hotspot = getHotspotByBlstId(blstId);
|
|
if (!hotspot) {
|
|
warning("Could not find hotspot with blstId %d", blstId);
|
|
return;
|
|
}
|
|
|
|
hotspot->setRect(position);
|
|
}
|
|
|
|
void RivenCard::addMenuHotspot(uint16 blstId, const Common::Rect &position, uint16 index,
|
|
uint16 externalCommandNameId, const char *externalCommandName) {
|
|
RivenHotspot *existingHotspot = getHotspotByBlstId(blstId);
|
|
if (existingHotspot) {
|
|
moveHotspot(blstId, position);
|
|
return; // Don't add the hotspot if it already exists
|
|
}
|
|
|
|
// Add the external command id => name mapping if it is missing
|
|
int16 existingCommandNameId = _vm->getStack()->getIdFromName(kExternalCommandNames, externalCommandName);
|
|
if (existingCommandNameId < 0) {
|
|
_vm->getStack()->registerName(kExternalCommandNames, externalCommandNameId, externalCommandName);
|
|
} else {
|
|
externalCommandNameId = existingCommandNameId;
|
|
}
|
|
|
|
uint16 patchData[] = {
|
|
blstId,
|
|
0xFFFF, // name
|
|
(uint16) position.left,
|
|
(uint16) position.top,
|
|
(uint16) position.right,
|
|
(uint16) position.bottom,
|
|
0, // u0
|
|
kRivenMainCursor, // cursor
|
|
index,
|
|
0xFFFF, // transition offset
|
|
0, // flags
|
|
2, // script count
|
|
|
|
kMouseDownScript, // script type
|
|
1, // command count
|
|
kRivenCommandRunExternal, // command type
|
|
2, // argument count
|
|
externalCommandNameId,
|
|
0, // external argument count
|
|
|
|
kMouseInsideScript, // script type
|
|
1, // command count
|
|
kRivenCommandChangeCursor, // command type
|
|
1, // argument count
|
|
kRivenOpenHandCursor // cursor
|
|
};
|
|
|
|
// Script data is expected to be in big endian
|
|
for (uint i = 0; i < ARRAYSIZE(patchData); i++) {
|
|
patchData[i] = TO_BE_16(patchData[i]);
|
|
}
|
|
|
|
// Add the new hotspot to the existing ones
|
|
Common::MemoryReadStream patchStream((const byte *)(patchData), ARRAYSIZE(patchData) * sizeof(uint16));
|
|
RivenHotspot *newHotspot = new RivenHotspot(_vm, &patchStream);
|
|
_hotspots.push_back(newHotspot);
|
|
}
|
|
|
|
void RivenCard::enter(bool unkMovies) {
|
|
setCurrentCardVariable();
|
|
|
|
_vm->_activatedPLST = false;
|
|
_vm->_activatedSLST = false;
|
|
|
|
_vm->_gfx->beginScreenUpdate();
|
|
runScript(kCardLoadScript);
|
|
defaultLoadScript();
|
|
|
|
initializeZipMode();
|
|
_vm->_gfx->applyScreenUpdate(true);
|
|
|
|
if (_vm->_showHotspots) {
|
|
drawHotspotRects();
|
|
}
|
|
|
|
runScript(kCardEnterScript);
|
|
}
|
|
|
|
void RivenCard::initializeZipMode() {
|
|
if (_zipModePlace) {
|
|
_vm->addZipVisitedCard(_id, _name);
|
|
}
|
|
|
|
// Check if a zip mode hotspot is enabled by checking the name/id against the ZIPS records.
|
|
for (uint32 i = 0; i < _hotspots.size(); i++) {
|
|
if (_hotspots[i]->isZip()) {
|
|
if (_vm->_vars["azip"] != 0) {
|
|
// Check if a zip mode hotspot is enabled by checking the name/id against the ZIPS records.
|
|
Common::String hotspotName = _hotspots[i]->getName();
|
|
bool visited = _vm->isZipVisitedCard(hotspotName);
|
|
|
|
_hotspots[i]->enable(visited);
|
|
} else // Disable the hotspot if zip mode is disabled
|
|
_hotspots[i]->enable(false);
|
|
}
|
|
}
|
|
}
|
|
|
|
RivenScriptPtr RivenCard::getScript(uint16 scriptType) const {
|
|
for (uint16 i = 0; i < _scripts.size(); i++)
|
|
if (_scripts[i].type == scriptType) {
|
|
return _scripts[i].script;
|
|
}
|
|
|
|
return RivenScriptPtr();
|
|
}
|
|
|
|
void RivenCard::runScript(uint16 scriptType) {
|
|
RivenScriptPtr script = getScript(scriptType);
|
|
_vm->_scriptMan->runScript(script, false);
|
|
}
|
|
|
|
uint16 RivenCard::getId() const {
|
|
return _id;
|
|
}
|
|
|
|
void RivenCard::defaultLoadScript() {
|
|
// Activate the first picture list if none have been activated
|
|
if (!_vm->_activatedPLST)
|
|
drawPicture(1);
|
|
|
|
// Activate the first sound list if none have been activated
|
|
if (!_vm->_activatedSLST)
|
|
playSound(1);
|
|
}
|
|
|
|
void RivenCard::loadCardPictureList(uint16 id) {
|
|
Common::SeekableReadStream* plst = _vm->getResource(ID_PLST, id);
|
|
uint16 recordCount = plst->readUint16BE();
|
|
_pictureList.resize(recordCount);
|
|
|
|
for (uint16 i = 0; i < recordCount; i++) {
|
|
Picture &picture = _pictureList[i];
|
|
picture.index = plst->readUint16BE();
|
|
picture.id = plst->readUint16BE();
|
|
picture.rect.left = plst->readUint16BE();
|
|
picture.rect.top = plst->readUint16BE();
|
|
picture.rect.right = plst->readUint16BE();
|
|
picture.rect.bottom = plst->readUint16BE();
|
|
}
|
|
|
|
delete plst;
|
|
}
|
|
|
|
void RivenCard::drawPicture(uint16 index, bool queue) {
|
|
if (index > 0 && index <= _pictureList.size()) {
|
|
RivenScriptPtr script = _vm->_scriptMan->createScriptFromData(1, kRivenCommandActivatePLST, 1, index);
|
|
_vm->_scriptMan->runScript(script, queue);
|
|
}
|
|
}
|
|
|
|
RivenCard::Picture RivenCard::getPicture(uint16 index) const {
|
|
for (uint16 i = 0; i < _pictureList.size(); i++) {
|
|
if (_pictureList[i].index == index) {
|
|
return _pictureList[i];
|
|
}
|
|
}
|
|
|
|
error("Could not find picture %d in card %d", index, _id);
|
|
}
|
|
|
|
void RivenCard::loadCardSoundList(uint16 id) {
|
|
Common::SeekableReadStream *slstStream = _vm->getResource(ID_SLST, id);
|
|
|
|
uint16 recordCount = slstStream->readUint16BE();
|
|
_soundList.resize(recordCount);
|
|
|
|
for (uint16 i = 0; i < recordCount; i++) {
|
|
SLSTRecord &slstRecord = _soundList[i];
|
|
slstRecord.index = slstStream->readUint16BE();
|
|
|
|
uint16 soundCount = slstStream->readUint16BE();
|
|
|
|
slstRecord.soundIds.resize(soundCount);
|
|
for (uint16 j = 0; j < soundCount; j++)
|
|
slstRecord.soundIds[j] = slstStream->readUint16BE();
|
|
|
|
slstRecord.fadeFlags = slstStream->readUint16BE();
|
|
slstRecord.loop = slstStream->readUint16BE();
|
|
slstRecord.globalVolume = slstStream->readUint16BE();
|
|
slstRecord.u0 = slstStream->readUint16BE(); // Unknown
|
|
|
|
if (slstRecord.u0 > 1)
|
|
warning("slstRecord.u0: %d non-boolean", slstRecord.u0);
|
|
|
|
slstRecord.suspend = slstStream->readUint16BE();
|
|
|
|
if (slstRecord.suspend != 0)
|
|
warning("slstRecord.suspend: %d non-zero", slstRecord.suspend);
|
|
|
|
slstRecord.volumes.resize(soundCount);
|
|
slstRecord.balances.resize(soundCount);
|
|
slstRecord.u2.resize(soundCount);
|
|
|
|
for (uint16 j = 0; j < soundCount; j++)
|
|
slstRecord.volumes[j] = slstStream->readUint16BE();
|
|
|
|
for (uint16 j = 0; j < soundCount; j++)
|
|
slstRecord.balances[j] = slstStream->readSint16BE(); // negative = left, 0 = center, positive = right
|
|
|
|
for (uint16 j = 0; j < soundCount; j++) {
|
|
slstRecord.u2[j] = slstStream->readUint16BE(); // Unknown
|
|
|
|
if (slstRecord.u2[j] != 255 && slstRecord.u2[j] != 256)
|
|
warning("slstRecord.u2[%d]: %d not 255 or 256", j, slstRecord.u2[j]);
|
|
}
|
|
}
|
|
|
|
delete slstStream;
|
|
}
|
|
|
|
void RivenCard::playSound(uint16 index, bool queue) {
|
|
if (index > 0 && index <= _soundList.size()) {
|
|
RivenScriptPtr script = _vm->_scriptMan->createScriptFromData(1, kRivenCommandActivateSLST, 1, index);
|
|
_vm->_scriptMan->runScript(script, queue);
|
|
}
|
|
}
|
|
|
|
SLSTRecord RivenCard::getSound(uint16 index) const {
|
|
for (uint16 i = 0; i < _soundList.size(); i++) {
|
|
if (_soundList[i].index == index) {
|
|
return _soundList[i];
|
|
}
|
|
}
|
|
|
|
error("Could not find sound %d in card %d", index, _id);
|
|
}
|
|
|
|
void RivenCard::overrideSound(uint16 index, uint16 withIndex) {
|
|
_soundList[index].soundIds = _soundList[withIndex].soundIds;
|
|
}
|
|
|
|
void RivenCard::loadHotspots(uint16 id) {
|
|
Common::SeekableReadStream *inStream = _vm->getResource(ID_HSPT, id);
|
|
|
|
uint16 hotspotCount = inStream->readUint16BE();
|
|
_hotspots.resize(hotspotCount);
|
|
|
|
uint32 globalId = _vm->getStack()->getCardGlobalId(id);
|
|
for (uint16 i = 0; i < hotspotCount; i++) {
|
|
_hotspots[i] = new RivenHotspot(_vm, inStream);
|
|
_hotspots[i]->applyPropertiesPatches(globalId);
|
|
_hotspots[i]->applyScriptPatches(globalId);
|
|
}
|
|
|
|
delete inStream;
|
|
}
|
|
|
|
void RivenCard::drawHotspotRects() {
|
|
for (uint16 i = 0; i < _hotspots.size(); i++)
|
|
_vm->_gfx->drawRect(_hotspots[i]->getRect(), _hotspots[i]->isEnabled());
|
|
}
|
|
|
|
RivenHotspot *RivenCard::getHotspotContainingPoint(const Common::Point &point) const {
|
|
RivenHotspot *hotspot = nullptr;
|
|
for (uint16 i = 0; i < _hotspots.size(); i++)
|
|
if (_hotspots[i]->isEnabled() && _hotspots[i]->containsPoint(point)) {
|
|
hotspot = _hotspots[i];
|
|
}
|
|
|
|
return hotspot;
|
|
}
|
|
|
|
Common::Array<RivenHotspot *> RivenCard::getHotspots() const {
|
|
return _hotspots;
|
|
}
|
|
|
|
RivenHotspot *RivenCard::getHotspotByName(const Common::String &name, bool optional) const {
|
|
int16 nameId = _vm->getStack()->getIdFromName(kHotspotNames, name);
|
|
|
|
for (uint i = 0; i < _hotspots.size(); i++) {
|
|
if (_hotspots[i]->getNameId() == nameId && nameId != -1) {
|
|
return _hotspots[i];
|
|
}
|
|
}
|
|
|
|
if (optional) {
|
|
return nullptr;
|
|
} else {
|
|
error("Card %d does not have an hotspot named %s", _id, name.c_str());
|
|
}
|
|
}
|
|
|
|
RivenHotspot *RivenCard::getHotspotByBlstId(const uint16 blstId) const {
|
|
for (uint i = 0; i < _hotspots.size(); i++) {
|
|
if (_hotspots[i]->getBlstId() == blstId) {
|
|
return _hotspots[i];
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
void RivenCard::loadCardHotspotEnableList(uint16 id) {
|
|
Common::SeekableReadStream *blst = _vm->getResource(ID_BLST, id);
|
|
|
|
uint16 recordCount = blst->readUint16BE();
|
|
_hotspotEnableList.resize(recordCount);
|
|
|
|
for (uint16 i = 0; i < recordCount; i++) {
|
|
HotspotEnableRecord &record = _hotspotEnableList[i];
|
|
record.index = blst->readUint16BE();
|
|
record.enabled = blst->readUint16BE();
|
|
record.hotspotId = blst->readUint16BE();
|
|
}
|
|
|
|
delete blst;
|
|
}
|
|
|
|
void RivenCard::activateHotspotEnableRecord(uint16 index) {
|
|
for (uint16 i = 0; i < _hotspotEnableList.size(); i++) {
|
|
const HotspotEnableRecord &record = _hotspotEnableList[i];
|
|
if (record.index == index) {
|
|
RivenHotspot *hotspot = getHotspotByBlstId(record.hotspotId);
|
|
hotspot->enable(record.enabled == 1);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void RivenCard::loadCardWaterEffectList(uint16 id) {
|
|
Common::SeekableReadStream *flst = _vm->getResource(ID_FLST, id);
|
|
|
|
uint16 recordCount = flst->readUint16BE();
|
|
_waterEffectList.resize(recordCount);
|
|
|
|
for (uint16 i = 0; i < recordCount; i++) {
|
|
WaterEffectRecord &record = _waterEffectList[i];
|
|
record.index = flst->readUint16BE();
|
|
record.sfxeId = flst->readUint16BE();
|
|
record.u0 = flst->readUint16BE();
|
|
|
|
if (record.u0 != 0) {
|
|
warning("FLST u0 non-zero");
|
|
}
|
|
}
|
|
|
|
delete flst;
|
|
}
|
|
|
|
void RivenCard::activateWaterEffect(uint16 index) {
|
|
for (uint16 i = 0; i < _waterEffectList.size(); i++) {
|
|
const WaterEffectRecord &record = _waterEffectList[i];
|
|
if (record.index == index) {
|
|
_vm->_gfx->scheduleWaterEffect(record.sfxeId);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
RivenHotspot *RivenCard::getCurHotspot() const {
|
|
return _hoveredHotspot;
|
|
}
|
|
|
|
RivenScriptPtr RivenCard::onMouseDown(const Common::Point &mouse) {
|
|
RivenScriptPtr script = onMouseMove(mouse);
|
|
updateMouseCursor();
|
|
|
|
_pressedHotspot = _hoveredHotspot;
|
|
|
|
if (_pressedHotspot) {
|
|
script += _pressedHotspot->getScript(kMouseDownScript);
|
|
}
|
|
|
|
return script;
|
|
}
|
|
|
|
RivenScriptPtr RivenCard::onMouseUp(const Common::Point &mouse) {
|
|
RivenScriptPtr script = onMouseMove(mouse);
|
|
|
|
if (_pressedHotspot && _pressedHotspot == _hoveredHotspot) {
|
|
script += _pressedHotspot->getScript(kMouseUpScript);
|
|
}
|
|
|
|
_pressedHotspot = nullptr;
|
|
|
|
return script;
|
|
}
|
|
|
|
RivenScriptPtr RivenCard::onMouseMove(const Common::Point &mouse) {
|
|
RivenHotspot *hotspot = getHotspotContainingPoint(mouse);
|
|
|
|
RivenScriptPtr script = RivenScriptPtr(new RivenScript());
|
|
|
|
// Detect hotspot exit
|
|
if (_hoveredHotspot && (!hotspot || hotspot != _hoveredHotspot)) {
|
|
script += _hoveredHotspot->getScript(kMouseLeaveScript);
|
|
}
|
|
|
|
// Detect hotspot entry
|
|
if (hotspot && hotspot != _hoveredHotspot) {
|
|
_hoveredHotspot = hotspot;
|
|
script += _hoveredHotspot->getScript(kMouseEnterScript);
|
|
}
|
|
|
|
if (!hotspot) {
|
|
_hoveredHotspot = nullptr;
|
|
}
|
|
|
|
return script;
|
|
}
|
|
|
|
RivenScriptPtr RivenCard::onMouseDragUpdate() {
|
|
RivenScriptPtr script;
|
|
if (_pressedHotspot) {
|
|
script = _pressedHotspot->getScript(kMouseDragScript);
|
|
}
|
|
|
|
return script;
|
|
}
|
|
|
|
RivenScriptPtr RivenCard::onFrame() {
|
|
return getScript(kCardFrameScript);
|
|
}
|
|
|
|
RivenScriptPtr RivenCard::onMouseUpdate() {
|
|
RivenScriptPtr script(new RivenScript());
|
|
|
|
if (_hoveredHotspot) {
|
|
script += _hoveredHotspot->getScript(kMouseInsideScript);
|
|
}
|
|
|
|
if (script->empty()) {
|
|
updateMouseCursor();
|
|
}
|
|
|
|
// Clear the pressed hotspot, in case we missed the mouse up event
|
|
// because we were running a script when it fired.
|
|
if (_pressedHotspot && _pressedHotspot == _hoveredHotspot) {
|
|
script += _pressedHotspot->getScript(kMouseUpScript);
|
|
}
|
|
_pressedHotspot = nullptr;
|
|
|
|
return script;
|
|
}
|
|
|
|
void RivenCard::updateMouseCursor() {
|
|
uint16 cursor;
|
|
if (_hoveredHotspot) {
|
|
cursor = _hoveredHotspot->getMouseCursor();
|
|
} else {
|
|
cursor = kRivenMainCursor;
|
|
}
|
|
|
|
_vm->_cursor->setCursor(cursor);
|
|
}
|
|
|
|
void RivenCard::leave() {
|
|
RivenScriptPtr script(new RivenScript());
|
|
|
|
if (_pressedHotspot) {
|
|
script += _pressedHotspot->getScript(kMouseUpScript);
|
|
_pressedHotspot = nullptr;
|
|
}
|
|
|
|
if (_hoveredHotspot) {
|
|
script += _hoveredHotspot->getScript(kMouseLeaveScript);
|
|
_hoveredHotspot = nullptr;
|
|
}
|
|
|
|
script += getScript(kCardLeaveScript);
|
|
|
|
_vm->_scriptMan->runScript(script, false);
|
|
}
|
|
|
|
void RivenCard::setCurrentCardVariable() {
|
|
_vm->_vars["currentcardid"] = _id;
|
|
}
|
|
|
|
void RivenCard::dump() const {
|
|
debug("== Card ==");
|
|
debug("id: %d", _id);
|
|
if (_name >= 0) {
|
|
debug("name: %s", _vm->getStack()->getName(kCardNames, _name).c_str());
|
|
} else {
|
|
debug("name: [no name]");
|
|
}
|
|
debug("zipModePlace: %d", _zipModePlace);
|
|
debug("globalId: %x", _vm->getStack()->getCardGlobalId(_id));
|
|
debugN("\n");
|
|
|
|
for (uint i = 0; i < _scripts.size(); i++) {
|
|
debug("== Script %d ==", i);
|
|
debug("type: %s", RivenScript::getTypeName(_scripts[i].type));
|
|
_scripts[i].script->dumpScript(0);
|
|
debugN("\n");
|
|
}
|
|
|
|
for (uint i = 0; i < _hotspots.size(); i++) {
|
|
debug("== Hotspot %d ==", i);
|
|
_hotspots[i]->dump();
|
|
}
|
|
|
|
for (uint i = 0; i < _pictureList.size(); i++) {
|
|
const Common::Rect &rect = _pictureList[i].rect;
|
|
debug("== Picture %d ==", _pictureList[i].index);
|
|
debug("pictureId: %d", _pictureList[i].id);
|
|
debug("rect: (%d, %d, %d, %d)", rect.left, rect.top, rect.right, rect.bottom);
|
|
debugN("\n");
|
|
}
|
|
|
|
for (uint i = 0; i < _waterEffectList.size(); i++) {
|
|
debug("== Effect %d ==", _waterEffectList[i].index);
|
|
debug("sfxeId: %d", _waterEffectList[i].sfxeId);
|
|
debug("u0: %d", _waterEffectList[i].u0);
|
|
debugN("\n");
|
|
}
|
|
|
|
for (uint i = 0; i < _hotspotEnableList.size(); i++) {
|
|
debug("== Hotspot enable %d ==", _hotspotEnableList[i].index);
|
|
debug("hotspotId: %d", _hotspotEnableList[i].hotspotId);
|
|
debug("enabled: %d", _hotspotEnableList[i].enabled);
|
|
debugN("\n");
|
|
}
|
|
|
|
for (uint i = 0; i < _soundList.size(); i++) {
|
|
debug("== Ambient sound list %d ==", _soundList[i].index);
|
|
debug("globalVolume: %d", _soundList[i].globalVolume);
|
|
debug("fadeFlags: %d", _soundList[i].fadeFlags);
|
|
debug("loop: %d", _soundList[i].loop);
|
|
debug("suspend: %d", _soundList[i].suspend);
|
|
debug("u0: %d", _soundList[i].u0);
|
|
for (uint j = 0; j < _soundList[i].soundIds.size(); j++) {
|
|
debug("sound[%d].id: %d", j, _soundList[i].soundIds[j]);
|
|
debug("sound[%d].volume: %d", j, _soundList[i].volumes[j]);
|
|
debug("sound[%d].balance: %d", j, _soundList[i].balances[j]);
|
|
debug("sound[%d].u2: %d", j, _soundList[i].u2[j]);
|
|
}
|
|
debugN("\n");
|
|
}
|
|
|
|
for (uint i = 0; i < _movieList.size(); i++) {
|
|
debug("== Movie %d ==", _movieList[i].index);
|
|
debug("movieID: %d", _movieList[i].movieID);
|
|
debug("playbackSlot: %d", _movieList[i].playbackSlot);
|
|
debug("left: %d", _movieList[i].left);
|
|
debug("top: %d", _movieList[i].top);
|
|
debug("lowBoundTime: %d", _movieList[i].lowBoundTime);
|
|
debug("startTime: %d", _movieList[i].startTime);
|
|
debug("highBoundTime: %d", _movieList[i].highBoundTime);
|
|
debug("loop: %d", _movieList[i].loop);
|
|
debug("volume: %d", _movieList[i].volume);
|
|
debug("rate: %d", _movieList[i].rate);
|
|
debugN("\n");
|
|
}
|
|
}
|
|
|
|
void RivenCard::loadCardMovieList(uint16 id) {
|
|
Common::SeekableReadStream *mlstStream = _vm->getResource(ID_MLST, id);
|
|
|
|
uint16 recordCount = mlstStream->readUint16BE();
|
|
_movieList.resize(recordCount);
|
|
|
|
for (uint16 i = 0; i < recordCount; i++) {
|
|
MLSTRecord &mlstRecord = _movieList[i];
|
|
mlstRecord.index = mlstStream->readUint16BE();
|
|
mlstRecord.movieID = mlstStream->readUint16BE();
|
|
mlstRecord.playbackSlot = mlstStream->readUint16BE();
|
|
mlstRecord.left = mlstStream->readUint16BE();
|
|
mlstRecord.top = mlstStream->readUint16BE();
|
|
mlstRecord.lowBoundTime = mlstStream->readUint16BE();
|
|
mlstRecord.startTime = mlstStream->readUint16BE();
|
|
mlstRecord.highBoundTime = mlstStream->readUint16BE();
|
|
mlstRecord.loop = mlstStream->readUint16BE();
|
|
mlstRecord.volume = mlstStream->readUint16BE();
|
|
mlstRecord.rate = mlstStream->readUint16BE();
|
|
|
|
if (mlstRecord.lowBoundTime != 0)
|
|
warning("lowBoundTime in MLST not 0");
|
|
|
|
if (mlstRecord.startTime != 0)
|
|
warning("startTime in MLST not 0");
|
|
|
|
if (mlstRecord.highBoundTime != 0xFFFF)
|
|
warning("highBoundTime in MLST not 0xFFFF");
|
|
|
|
if (mlstRecord.rate != 1)
|
|
warning("mlstRecord.rate not 1");
|
|
}
|
|
|
|
delete mlstStream;
|
|
}
|
|
|
|
MLSTRecord RivenCard::getMovie(uint16 index) const {
|
|
for (uint16 i = 0; i < _movieList.size(); i++) {
|
|
if (_movieList[i].index == index) {
|
|
return _movieList[i];
|
|
}
|
|
}
|
|
|
|
error("Could not find movie %d in card %d", index, _id);
|
|
}
|
|
|
|
void RivenCard::playMovie(uint16 index, bool queue) {
|
|
if (index > 0 && index <= _movieList.size()) {
|
|
RivenScriptPtr script = _vm->_scriptMan->createScriptFromData(1, kRivenCommandActivateMLSTAndPlay, 1, index);
|
|
_vm->_scriptMan->runScript(script, queue);
|
|
}
|
|
}
|
|
|
|
RivenScriptPtr RivenCard::onKeyAction(RivenAction keyAction) {
|
|
if (_pressedHotspot) {
|
|
return RivenScriptPtr(new RivenScript());
|
|
}
|
|
|
|
static const char *forwardNames[] = {
|
|
"forward", "forward1", "forward2", "forward3",
|
|
"opendoor", "openhatch", "opentrap", "opengate", "opengrate",
|
|
"open", "door", "drop", "go", "enterprison", "exit",
|
|
"forwardleft", "forwardright", nullptr
|
|
};
|
|
|
|
static const char *forwardLeftNames [] = { "forwardleft", nullptr };
|
|
static const char *forwardRightNames[] = { "forwardright", nullptr };
|
|
static const char *leftNames [] = { "left", "afl", "prevpage", nullptr };
|
|
static const char *rightNames [] = { "right", "afr", "nextpage", nullptr };
|
|
static const char *backNames [] = { "back", nullptr };
|
|
static const char *upNames [] = { "up", nullptr };
|
|
static const char *downNames [] = { "down", nullptr };
|
|
|
|
const char **hotspotNames = nullptr;
|
|
switch (keyAction) {
|
|
case kRivenActionMoveForward:
|
|
hotspotNames = forwardNames;
|
|
break;
|
|
case kRivenActionMoveForwardLeft:
|
|
hotspotNames = forwardLeftNames;
|
|
break;
|
|
case kRivenActionMoveForwardRight:
|
|
hotspotNames = forwardRightNames;
|
|
break;
|
|
case kRivenActionMoveLeft:
|
|
hotspotNames = leftNames;
|
|
break;
|
|
case kRivenActionMoveRight:
|
|
hotspotNames = rightNames;
|
|
break;
|
|
case kRivenActionMoveBack:
|
|
hotspotNames = backNames;
|
|
break;
|
|
case kRivenActionLookUp:
|
|
hotspotNames = upNames;
|
|
break;
|
|
case kRivenActionLookDown:
|
|
hotspotNames = downNames;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (!hotspotNames) {
|
|
return RivenScriptPtr(new RivenScript());
|
|
}
|
|
|
|
RivenHotspot *directionHotspot = findEnabledHotspotByName(hotspotNames);
|
|
if (!directionHotspot) {
|
|
return RivenScriptPtr(new RivenScript());
|
|
}
|
|
|
|
_hoveredHotspot = directionHotspot;
|
|
|
|
RivenScriptPtr clickScript = directionHotspot->getScript(kMouseDownScript);
|
|
if (!clickScript || clickScript->empty()) {
|
|
clickScript = directionHotspot->getScript(kMouseUpScript);
|
|
}
|
|
if (!clickScript || clickScript->empty()) {
|
|
clickScript = RivenScriptPtr(new RivenScript());
|
|
}
|
|
|
|
return clickScript;
|
|
}
|
|
|
|
RivenHotspot *RivenCard::findEnabledHotspotByName(const char **names) const {
|
|
for (uint i = 0; names[i] != nullptr; i++) {
|
|
RivenHotspot *hotspot = getHotspotByName(names[i], true);
|
|
if (hotspot && hotspot->isEnabled()) {
|
|
return hotspot;
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
RivenHotspot::RivenHotspot(MohawkEngine_Riven *vm, Common::ReadStream *stream) :
|
|
_vm(vm) {
|
|
loadFromStream(stream);
|
|
}
|
|
|
|
void RivenHotspot::loadFromStream(Common::ReadStream *stream) {
|
|
_flags = kFlagEnabled;
|
|
|
|
_blstID = stream->readUint16BE();
|
|
_nameResource = stream->readSint16BE();
|
|
|
|
int16 left = stream->readSint16BE();
|
|
int16 top = stream->readSint16BE();
|
|
int16 right = stream->readSint16BE();
|
|
int16 bottom = stream->readSint16BE();
|
|
|
|
// Riven has some invalid rects, disable them here
|
|
// Known weird hotspots:
|
|
// - tspit 371 (DVD: 377), hotspot 4
|
|
if (left >= right || top >= bottom) {
|
|
warning("Invalid hotspot: (%d, %d, %d, %d)", left, top, right, bottom);
|
|
left = top = right = bottom = 0;
|
|
enable(false);
|
|
}
|
|
|
|
_rect = Common::Rect(left, top, right, bottom);
|
|
|
|
_u0 = stream->readUint16BE();
|
|
_mouseCursor = stream->readUint16BE();
|
|
_index = stream->readUint16BE();
|
|
_transitionOffset = stream->readSint16BE();
|
|
_flags |= stream->readUint16BE();
|
|
|
|
// Read in the scripts now
|
|
_scripts = _vm->_scriptMan->readScripts(stream);
|
|
}
|
|
|
|
RivenScriptPtr RivenHotspot::getScript(uint16 scriptType) const {
|
|
for (uint16 i = 0; i < _scripts.size(); i++)
|
|
if (_scripts[i].type == scriptType) {
|
|
return _scripts[i].script;
|
|
}
|
|
|
|
return RivenScriptPtr();
|
|
}
|
|
|
|
void RivenHotspot::applyPropertiesPatches(uint32 cardGlobalId) {
|
|
// In Jungle island, one of the bridge hotspots does not have a name
|
|
// This breaks keyboard navigation. Set the proper name.
|
|
if (cardGlobalId == 0x214a0 && _blstID == 9) {
|
|
_nameResource = _vm->getStack()->getIdFromName(kHotspotNames, "forward");
|
|
debugC(kRivenDebugPatches, "Applied missing hotspot name patch to card %x", cardGlobalId);
|
|
}
|
|
|
|
// In the lab in Book Making island the card showing one of the doors has
|
|
// two "forward" hotspots. One of them goes backwards. Disable it, and make sure
|
|
// it cannot be found by the keyboard navigation code.
|
|
if (cardGlobalId == 0x1fa79 && _blstID == 3) {
|
|
enable(false);
|
|
_nameResource = -1;
|
|
debugC(kRivenDebugPatches, "Applied disable buggy forward hotspot to card %x", cardGlobalId);
|
|
}
|
|
|
|
// On Temple Island, in front of the back door to the rotating room,
|
|
// change the name of the hotspot to look at the bottom of the door to
|
|
// "down" instead of "forwardleft". That way the keyboard navigation
|
|
// does not spoil that you can go below the door.
|
|
// Also make sure the forward keyboard action plays the try to open
|
|
// door animation.
|
|
if (cardGlobalId == 0x87ac && _blstID == 10) {
|
|
_nameResource = _vm->getStack()->getIdFromName(kHotspotNames, "down");
|
|
debugC(kRivenDebugPatches, "Applied change hotspot name to 'down' patch to card %x", cardGlobalId);
|
|
}
|
|
if (cardGlobalId == 0x87ac && _blstID == 12) {
|
|
_nameResource = _vm->getStack()->getIdFromName(kHotspotNames, "opendoor");
|
|
debugC(kRivenDebugPatches, "Applied change hotspot name to 'opendoor' patch to card %x", cardGlobalId);
|
|
}
|
|
}
|
|
|
|
void RivenHotspot::applyScriptPatches(uint32 cardGlobalId) {
|
|
for (uint16 i = 0; i < _scripts.size(); i++) {
|
|
_scripts[i].script->applyCardPatches(_vm, cardGlobalId, _scripts[i].type, _blstID);
|
|
}
|
|
}
|
|
|
|
bool RivenHotspot::isEnabled() const {
|
|
return (_flags & kFlagEnabled) != 0;
|
|
}
|
|
|
|
void RivenHotspot::enable(bool e) {
|
|
if (e) {
|
|
_flags |= kFlagEnabled;
|
|
} else {
|
|
_flags &= ~kFlagEnabled;
|
|
}
|
|
}
|
|
|
|
bool RivenHotspot::isZip() const {
|
|
return (_flags & kFlagZip) != 0;
|
|
}
|
|
|
|
Common::Rect RivenHotspot::getRect() const {
|
|
return _rect;
|
|
}
|
|
|
|
bool RivenHotspot::containsPoint(const Common::Point &point) const {
|
|
return _rect.contains(point);
|
|
}
|
|
|
|
uint16 RivenHotspot::getMouseCursor() const {
|
|
return _mouseCursor;
|
|
}
|
|
|
|
Common::String RivenHotspot::getName() const {
|
|
if (_nameResource < 0)
|
|
return Common::String();
|
|
|
|
return _vm->getStack()->getName(kHotspotNames, _nameResource);
|
|
}
|
|
|
|
uint16 RivenHotspot::getIndex() const {
|
|
return _index;
|
|
}
|
|
|
|
uint16 RivenHotspot::getBlstId() const {
|
|
return _blstID;
|
|
}
|
|
|
|
void RivenHotspot::setRect(const Common::Rect &rect) {
|
|
_rect = rect;
|
|
}
|
|
|
|
int16 RivenHotspot::getNameId() const {
|
|
return _nameResource;
|
|
}
|
|
|
|
int16 RivenHotspot::getTransitionOffset() const {
|
|
return _transitionOffset;
|
|
}
|
|
|
|
void RivenHotspot::dump() const {
|
|
debug("index: %d", _index);
|
|
debug("blstId: %d", _blstID);
|
|
debug("name: %s", getName().c_str());
|
|
debug("rect: (%d, %d, %d, %d)", _rect.left, _rect.top, _rect.right, _rect.bottom);
|
|
debug("flags: %d", _flags);
|
|
debug("mouseCursor: %d", _mouseCursor);
|
|
debug("transitionOffset: %d", _transitionOffset);
|
|
debug("u0: %d", _u0);
|
|
debugN("\n");
|
|
|
|
for (uint i = 0; i < _scripts.size(); i++) {
|
|
debug("=== Hotspot script %d ===", i);
|
|
debug("type: %s", RivenScript::getTypeName(_scripts[i].type));
|
|
_scripts[i].script->dumpScript(0);
|
|
debugN("\n");
|
|
}
|
|
}
|
|
|
|
} // End of namespace Mohawk
|