2009-12-29 23:18:24 +00:00
|
|
|
/* 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 2
|
|
|
|
* of the License, or (at your option) any later version.
|
2014-02-18 02:34:22 +01:00
|
|
|
*
|
2009-12-29 23:18:24 +00:00
|
|
|
* 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.
|
2014-02-18 02:34:22 +01:00
|
|
|
*
|
2009-12-29 23:18:24 +00:00
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2010-11-25 04:49:11 +00:00
|
|
|
#include "mohawk/cursors.h"
|
2009-12-29 23:18:24 +00:00
|
|
|
#include "mohawk/riven.h"
|
|
|
|
#include "mohawk/riven_external.h"
|
2012-03-10 13:50:27 -05:00
|
|
|
#include "mohawk/riven_graphics.h"
|
2009-12-29 23:18:24 +00:00
|
|
|
#include "mohawk/riven_scripts.h"
|
2016-08-08 07:35:55 +02:00
|
|
|
#include "mohawk/riven_sound.h"
|
2010-05-23 18:33:55 +00:00
|
|
|
#include "mohawk/video.h"
|
2009-12-29 23:18:24 +00:00
|
|
|
|
2011-03-22 20:31:54 -04:00
|
|
|
#include "common/memstream.h"
|
2009-12-29 23:18:24 +00:00
|
|
|
#include "common/stream.h"
|
2011-04-24 11:34:27 +03:00
|
|
|
#include "common/system.h"
|
2009-12-29 23:18:24 +00:00
|
|
|
|
|
|
|
namespace Mohawk {
|
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
static void printTabs(byte tabs) {
|
|
|
|
for (byte i = 0; i < tabs; i++)
|
|
|
|
debugN("\t");
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
RivenScriptManager::RivenScriptManager(MohawkEngine_Riven *vm) {
|
|
|
|
_vm = vm;
|
|
|
|
_storedMovieOpcode.time = 0;
|
|
|
|
_storedMovieOpcode.id = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
RivenScriptManager::~RivenScriptManager() {
|
|
|
|
clearStoredMovieOpcode();
|
|
|
|
}
|
|
|
|
|
2016-08-02 19:43:15 +02:00
|
|
|
RivenScriptPtr RivenScriptManager::readScript(Common::ReadStream *stream) {
|
2016-08-02 21:04:34 +02:00
|
|
|
RivenScriptPtr script = RivenScriptPtr(new RivenScript());
|
2016-08-02 18:56:55 +02:00
|
|
|
|
|
|
|
uint16 commandCount = stream->readUint16BE();
|
|
|
|
|
|
|
|
for (uint16 i = 0; i < commandCount; i++) {
|
|
|
|
RivenCommand *command = readCommand(stream);
|
|
|
|
script->addCommand(command);
|
|
|
|
}
|
|
|
|
|
|
|
|
return script;
|
|
|
|
}
|
|
|
|
|
|
|
|
RivenCommand *RivenScriptManager::readCommand(Common::ReadStream *stream) {
|
|
|
|
uint16 type = stream->readUint16BE();
|
|
|
|
|
|
|
|
if (type == 8) {
|
|
|
|
return RivenSwitchCommand::createFromStream(_vm, type, stream);
|
2009-12-29 23:18:24 +00:00
|
|
|
} else {
|
2016-08-02 18:56:55 +02:00
|
|
|
return RivenSimpleCommand::createFromStream(_vm, type, stream);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
RivenScriptList RivenScriptManager::readScripts(Common::ReadStream *stream) {
|
|
|
|
RivenScriptList scriptList;
|
|
|
|
|
|
|
|
uint16 scriptCount = stream->readUint16BE();
|
|
|
|
for (uint16 i = 0; i < scriptCount; i++) {
|
2016-08-02 19:43:15 +02:00
|
|
|
RivenTypedScript script;
|
|
|
|
script.type = stream->readUint16BE();
|
|
|
|
script.script = readScript(stream);
|
2016-08-02 18:56:55 +02:00
|
|
|
scriptList.push_back(script);
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
return scriptList;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RivenScriptManager::stopAllScripts() {
|
|
|
|
// TODO: Restore
|
|
|
|
// for (uint32 i = 0; i < _currentScripts.size(); i++)
|
|
|
|
// _currentScripts[i]->stopRunning();
|
|
|
|
}
|
|
|
|
|
|
|
|
void RivenScriptManager::setStoredMovieOpcode(const StoredMovieOpcode &op) {
|
|
|
|
clearStoredMovieOpcode();
|
|
|
|
_storedMovieOpcode.script = op.script;
|
|
|
|
_storedMovieOpcode.id = op.id;
|
|
|
|
_storedMovieOpcode.time = op.time;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RivenScriptManager::runStoredMovieOpcode() {
|
|
|
|
if (_storedMovieOpcode.script) {
|
2016-08-02 21:43:22 +02:00
|
|
|
runScript(_storedMovieOpcode.script, false);
|
2016-08-02 18:56:55 +02:00
|
|
|
clearStoredMovieOpcode();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void RivenScriptManager::clearStoredMovieOpcode() {
|
|
|
|
_storedMovieOpcode.script = RivenScriptPtr();
|
|
|
|
_storedMovieOpcode.time = 0;
|
|
|
|
_storedMovieOpcode.id = 0;
|
|
|
|
}
|
|
|
|
|
2016-08-02 21:43:22 +02:00
|
|
|
void RivenScriptManager::runScript(const RivenScriptPtr &script, bool queue) {
|
|
|
|
if (!queue) {
|
|
|
|
script->run();
|
|
|
|
} else {
|
|
|
|
_queue.push_back(script);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-02 21:04:34 +02:00
|
|
|
RivenScript::RivenScript() {
|
2016-08-02 18:56:55 +02:00
|
|
|
_continueRunning = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
RivenScript::~RivenScript() {
|
|
|
|
for (uint i = 0; i < _commands.size(); i ++) {
|
|
|
|
delete _commands[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void RivenScript::dumpScript(const Common::StringArray &varNames, const Common::StringArray &xNames, byte tabs) {
|
|
|
|
for (uint16 i = 0; i < _commands.size(); i++) {
|
|
|
|
_commands[i]->dump(varNames, xNames, tabs);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-02 21:43:22 +02:00
|
|
|
void RivenScript::run() {
|
2016-08-02 18:56:55 +02:00
|
|
|
for (uint16 i = 0; i < _commands.size() && _continueRunning; i++) {
|
|
|
|
_commands[i]->execute();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void RivenScript::addCommand(RivenCommand *command) {
|
|
|
|
_commands.push_back(command);
|
|
|
|
}
|
|
|
|
|
|
|
|
RivenCommand::RivenCommand(MohawkEngine_Riven *vm) :
|
|
|
|
_vm(vm) {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
RivenCommand::~RivenCommand() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
RivenSimpleCommand::RivenSimpleCommand(MohawkEngine_Riven *vm, int type, const ArgumentArray &arguments) :
|
|
|
|
RivenCommand(vm),
|
|
|
|
_type(type),
|
|
|
|
_arguments(arguments) {
|
|
|
|
setupOpcodes();
|
|
|
|
}
|
|
|
|
|
|
|
|
RivenSimpleCommand::~RivenSimpleCommand() {
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
RivenSimpleCommand *RivenSimpleCommand::createFromStream(MohawkEngine_Riven *vm, int type, Common::ReadStream *stream) {
|
|
|
|
uint16 argCount = stream->readUint16BE();
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
Common::Array<uint16> arguments;
|
|
|
|
arguments.resize(argCount);
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
for (uint16 i = 0; i < argCount; i++) {
|
|
|
|
arguments[i] = stream->readUint16BE();
|
|
|
|
}
|
|
|
|
|
|
|
|
return new RivenSimpleCommand(vm, type, arguments);
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
#define OPCODE(x) { &RivenSimpleCommand::x, #x }
|
2009-12-29 23:18:24 +00:00
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::setupOpcodes() {
|
2009-12-29 23:18:24 +00:00
|
|
|
static const RivenOpcode riven_opcodes[] = {
|
|
|
|
// 0x00 (0 decimal)
|
|
|
|
OPCODE(empty),
|
|
|
|
OPCODE(drawBitmap),
|
|
|
|
OPCODE(switchCard),
|
|
|
|
OPCODE(playScriptSLST),
|
|
|
|
// 0x04 (4 decimal)
|
|
|
|
OPCODE(playSound),
|
|
|
|
OPCODE(empty), // Empty
|
|
|
|
OPCODE(empty), // Complex animation (not used)
|
|
|
|
OPCODE(setVariable),
|
|
|
|
// 0x08 (8 decimal)
|
|
|
|
OPCODE(mohawkSwitch),
|
|
|
|
OPCODE(enableHotspot),
|
|
|
|
OPCODE(disableHotspot),
|
|
|
|
OPCODE(empty), // Empty
|
|
|
|
// 0x0C (12 decimal)
|
2011-01-23 05:12:50 +00:00
|
|
|
OPCODE(stopSound),
|
2009-12-29 23:18:24 +00:00
|
|
|
OPCODE(changeCursor),
|
|
|
|
OPCODE(delay),
|
|
|
|
OPCODE(empty), // Empty
|
|
|
|
// 0x10 (16 decimal)
|
|
|
|
OPCODE(empty), // Empty
|
|
|
|
OPCODE(runExternalCommand),
|
|
|
|
OPCODE(transition),
|
|
|
|
OPCODE(refreshCard),
|
|
|
|
// 0x14 (20 decimal)
|
|
|
|
OPCODE(disableScreenUpdate),
|
|
|
|
OPCODE(enableScreenUpdate),
|
|
|
|
OPCODE(empty), // Empty
|
|
|
|
OPCODE(empty), // Empty
|
|
|
|
// 0x18 (24 decimal)
|
|
|
|
OPCODE(incrementVariable),
|
|
|
|
OPCODE(empty), // Empty
|
|
|
|
OPCODE(empty), // Empty
|
|
|
|
OPCODE(changeStack),
|
|
|
|
// 0x1C (28 decimal)
|
|
|
|
OPCODE(disableMovie),
|
|
|
|
OPCODE(disableAllMovies),
|
|
|
|
OPCODE(empty), // Set movie rate (not used)
|
|
|
|
OPCODE(enableMovie),
|
|
|
|
// 0x20 (32 decimal)
|
2011-01-18 20:30:16 +00:00
|
|
|
OPCODE(playMovieBlocking),
|
2009-12-29 23:18:24 +00:00
|
|
|
OPCODE(playMovie),
|
|
|
|
OPCODE(stopMovie),
|
|
|
|
OPCODE(empty), // Start a water effect (not used)
|
|
|
|
// 0x24 (36 decimal)
|
|
|
|
OPCODE(unk_36), // Unknown
|
|
|
|
OPCODE(fadeAmbientSounds),
|
2011-03-22 20:31:54 -04:00
|
|
|
OPCODE(storeMovieOpcode),
|
2009-12-29 23:18:24 +00:00
|
|
|
OPCODE(activatePLST),
|
|
|
|
// 0x28 (40 decimal)
|
|
|
|
OPCODE(activateSLST),
|
|
|
|
OPCODE(activateMLSTAndPlay),
|
|
|
|
OPCODE(empty), // Empty
|
|
|
|
OPCODE(activateBLST),
|
|
|
|
// 0x2C (44 decimal)
|
|
|
|
OPCODE(activateFLST),
|
|
|
|
OPCODE(zipMode),
|
|
|
|
OPCODE(activateMLST),
|
2010-07-09 16:53:20 +00:00
|
|
|
OPCODE(empty) // Activate an SLST with a volume parameter (not used)
|
2009-12-29 23:18:24 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
_opcodes = riven_opcodes;
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////
|
|
|
|
// Opcodes
|
|
|
|
////////////////////////////////
|
|
|
|
|
|
|
|
// Command 1: draw tBMP resource (tbmp_id, left, top, right, bottom, u0, u1, u2, u3)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::drawBitmap(uint16 op, uint16 argc, uint16 *argv) {
|
2010-06-02 15:26:35 +00:00
|
|
|
if (argc < 5) // Copy the image to the whole screen, ignoring the rest of the parameters
|
2009-12-29 23:18:24 +00:00
|
|
|
_vm->_gfx->copyImageToScreen(argv[0], 0, 0, 608, 392);
|
2011-01-19 15:22:39 +00:00
|
|
|
else // Copy the image to a certain part of the screen
|
2009-12-29 23:18:24 +00:00
|
|
|
_vm->_gfx->copyImageToScreen(argv[0], argv[1], argv[2], argv[3], argv[4]);
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-12-29 23:18:24 +00:00
|
|
|
// Now, update the screen
|
|
|
|
_vm->_gfx->updateScreen();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Command 2: go to card (card id)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::switchCard(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
_vm->changeToCard(argv[0]);
|
2010-06-02 15:26:35 +00:00
|
|
|
|
|
|
|
// WORKAROUND: If we changed card on a mouse down event,
|
|
|
|
// we want to ignore the next mouse up event so we don't
|
|
|
|
// change card when lifting the mouse on the next card.
|
2016-08-02 18:56:55 +02:00
|
|
|
// TODO: Restore
|
|
|
|
// if (_scriptType == kMouseDownScript)
|
|
|
|
// _vm->ignoreNextMouseUp();
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Command 3: play an SLST from the script
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::playScriptSLST(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
int offset = 0, j = 0;
|
2016-08-08 07:35:55 +02:00
|
|
|
uint16 soundCount = argv[offset++];
|
2009-12-29 23:18:24 +00:00
|
|
|
|
2016-08-08 07:35:55 +02:00
|
|
|
SLSTRecord slstRecord;
|
2009-12-29 23:18:24 +00:00
|
|
|
slstRecord.index = 0; // not set by the scripts, so we set it to 0
|
2016-08-08 07:35:55 +02:00
|
|
|
slstRecord.soundIds.resize(soundCount);
|
2009-12-29 23:18:24 +00:00
|
|
|
|
2016-08-08 07:35:55 +02:00
|
|
|
for (j = 0; j < soundCount; j++)
|
|
|
|
slstRecord.soundIds[j] = argv[offset++];
|
|
|
|
slstRecord.fadeFlags = argv[offset++];
|
2009-12-29 23:18:24 +00:00
|
|
|
slstRecord.loop = argv[offset++];
|
2016-08-08 07:35:55 +02:00
|
|
|
slstRecord.globalVolume = argv[offset++];
|
2010-01-25 01:39:44 +00:00
|
|
|
slstRecord.u0 = argv[offset++];
|
2016-08-08 07:35:55 +02:00
|
|
|
slstRecord.suspend = argv[offset++];
|
2009-12-29 23:18:24 +00:00
|
|
|
|
2016-08-08 07:35:55 +02:00
|
|
|
slstRecord.volumes.resize(soundCount);
|
|
|
|
slstRecord.balances.resize(soundCount);
|
|
|
|
slstRecord.u2.resize(soundCount);
|
2009-12-29 23:18:24 +00:00
|
|
|
|
2016-08-08 07:35:55 +02:00
|
|
|
for (j = 0; j < soundCount; j++)
|
2009-12-29 23:18:24 +00:00
|
|
|
slstRecord.volumes[j] = argv[offset++];
|
|
|
|
|
2016-08-08 07:35:55 +02:00
|
|
|
for (j = 0; j < soundCount; j++)
|
2009-12-29 23:18:24 +00:00
|
|
|
slstRecord.balances[j] = argv[offset++]; // negative = left, 0 = center, positive = right
|
|
|
|
|
2016-08-08 07:35:55 +02:00
|
|
|
for (j = 0; j < soundCount; j++)
|
2009-12-29 23:18:24 +00:00
|
|
|
slstRecord.u2[j] = argv[offset++]; // Unknown
|
|
|
|
|
|
|
|
// Play the requested sound list
|
|
|
|
_vm->_sound->playSLST(slstRecord);
|
|
|
|
}
|
|
|
|
|
2010-09-08 20:50:56 +00:00
|
|
|
// Command 4: play local tWAV resource (twav_id, volume, block)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::playSound(uint16 op, uint16 argc, uint16 *argv) {
|
2016-08-08 07:35:55 +02:00
|
|
|
uint16 volume = argv[1];
|
|
|
|
bool playOnDraw = argv[2] == 1;
|
2010-09-08 20:50:56 +00:00
|
|
|
|
2016-08-08 07:35:55 +02:00
|
|
|
_vm->_sound->playSound(argv[0], volume, playOnDraw);
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Command 7: set variable value (variable, value)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::setVariable(uint16 op, uint16 argc, uint16 *argv) {
|
2011-03-23 23:14:59 -04:00
|
|
|
_vm->getStackVar(argv[0]) = argv[1];
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Command 8: conditional branch
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::mohawkSwitch(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
// dummy function, this opcode does logic checking in processCommands()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Command 9: enable hotspot (blst_id)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::enableHotspot(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
for (uint16 i = 0; i < _vm->getHotspotCount(); i++) {
|
|
|
|
if (_vm->_hotspots[i].blstID == argv[0]) {
|
|
|
|
debug(2, "Enabling hotspot with BLST ID %d", argv[0]);
|
|
|
|
_vm->_hotspots[i].enabled = true;
|
|
|
|
}
|
|
|
|
}
|
2011-01-24 02:43:07 +00:00
|
|
|
|
|
|
|
// Recheck our current hotspot because it may have now changed
|
|
|
|
_vm->updateCurrentHotspot();
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Command 10: disable hotspot (blst_id)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::disableHotspot(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
for (uint16 i = 0; i < _vm->getHotspotCount(); i++) {
|
|
|
|
if (_vm->_hotspots[i].blstID == argv[0]) {
|
|
|
|
debug(2, "Disabling hotspot with BLST ID %d", argv[0]);
|
|
|
|
_vm->_hotspots[i].enabled = false;
|
|
|
|
}
|
|
|
|
}
|
2011-01-24 02:43:07 +00:00
|
|
|
|
|
|
|
// Recheck our current hotspot because it may have now changed
|
|
|
|
_vm->updateCurrentHotspot();
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
2011-01-23 05:12:50 +00:00
|
|
|
// Command 12: stop sounds (flags)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::stopSound(uint16 op, uint16 argc, uint16 *argv) {
|
2011-01-23 07:02:49 +00:00
|
|
|
// WORKAROUND: The Play Riven/Visit Riven/Start New Game buttons
|
|
|
|
// in the main menu call this function to stop ambient sounds
|
|
|
|
// after the change stack call to Temple Island. However, this
|
|
|
|
// would cause all ambient sounds not to play. An alternative
|
|
|
|
// fix would be to stop all scripts on a stack change, but this
|
|
|
|
// does fine for now.
|
2014-05-03 22:55:50 -04:00
|
|
|
if (_vm->getCurStack() == kStackTspit && (_vm->getCurCardRMAP() == 0x6e9a || _vm->getCurCardRMAP() == 0xfeeb))
|
2011-01-23 07:02:49 +00:00
|
|
|
return;
|
|
|
|
|
2011-01-23 05:12:50 +00:00
|
|
|
// The argument is a bitflag for the setting.
|
2011-06-14 10:46:48 -04:00
|
|
|
// bit 0 is normal sound stopping
|
2011-01-23 05:12:50 +00:00
|
|
|
// bit 1 is ambient sound stopping
|
|
|
|
// Having no flags set means clear all
|
2011-01-23 07:02:49 +00:00
|
|
|
if (argv[0] & 2 || argv[0] == 0)
|
|
|
|
_vm->_sound->stopAllSLST();
|
2011-01-23 05:12:50 +00:00
|
|
|
|
2011-06-14 10:46:48 -04:00
|
|
|
if (argv[0] & 1 || argv[0] == 0)
|
|
|
|
_vm->_sound->stopSound();
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Command 13: set mouse cursor (cursor_id)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::changeCursor(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
debug(2, "Change to cursor %d", argv[0]);
|
2010-11-25 04:49:11 +00:00
|
|
|
_vm->_cursor->setCursor(argv[0]);
|
2011-03-07 01:07:53 -05:00
|
|
|
_vm->_system->updateScreen();
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Command 14: pause script execution (delay in ms, u1)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::delay(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
debug(2, "Delay %dms", argv[0]);
|
|
|
|
if (argv[0] > 0)
|
2010-09-01 23:39:25 +00:00
|
|
|
_vm->delayAndUpdate(argv[0]);
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Command 17: call external command
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::runExternalCommand(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
_vm->_externalScriptHandler->runCommand(argc, argv);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Command 18: transition
|
2011-01-19 15:22:39 +00:00
|
|
|
// Note that this opcode has 1 or 5 parameters, depending on argc
|
2009-12-29 23:18:24 +00:00
|
|
|
// Parameter 0: transition type
|
|
|
|
// Parameters 1-4: transition rectangle
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::transition(uint16 op, uint16 argc, uint16 *argv) {
|
2010-02-26 08:14:33 +00:00
|
|
|
if (argc == 1)
|
2009-12-29 23:18:24 +00:00
|
|
|
_vm->_gfx->scheduleTransition(argv[0]);
|
2010-02-26 08:14:33 +00:00
|
|
|
else
|
2009-12-29 23:18:24 +00:00
|
|
|
_vm->_gfx->scheduleTransition(argv[0], Common::Rect(argv[1], argv[2], argv[3], argv[4]));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Command 19: reload card
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::refreshCard(uint16 op, uint16 argc, uint16 *argv) {
|
2010-02-26 08:14:33 +00:00
|
|
|
debug(2, "Refreshing card");
|
|
|
|
_vm->refreshCard();
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Command 20: disable screen update
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::disableScreenUpdate(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
debug(2, "Screen update disabled");
|
|
|
|
_vm->_gfx->_updatesEnabled = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Command 21: enable screen update
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::enableScreenUpdate(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
debug(2, "Screen update enabled");
|
|
|
|
_vm->_gfx->_updatesEnabled = true;
|
|
|
|
_vm->_gfx->updateScreen();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Command 24: increment variable (variable, value)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::incrementVariable(uint16 op, uint16 argc, uint16 *argv) {
|
2011-03-23 23:14:59 -04:00
|
|
|
_vm->getStackVar(argv[0]) += argv[1];
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
2011-03-22 20:31:54 -04:00
|
|
|
// Command 27: go to stack (stack name, code high, code low)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::changeStack(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
Common::String stackName = _vm->getName(StackNames, argv[0]);
|
|
|
|
int8 index = -1;
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-12-29 23:18:24 +00:00
|
|
|
for (byte i = 0; i < 8; i++)
|
2011-01-19 15:22:39 +00:00
|
|
|
if (_vm->getStackName(i).equalsIgnoreCase(stackName)) {
|
2009-12-29 23:18:24 +00:00
|
|
|
index = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (index == -1)
|
2011-01-19 15:22:39 +00:00
|
|
|
error ("'%s' is not a stack name!", stackName.c_str());
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-12-29 23:18:24 +00:00
|
|
|
_vm->changeToStack(index);
|
|
|
|
uint32 rmapCode = (argv[1] << 16) + argv[2];
|
|
|
|
uint16 cardID = _vm->matchRMAPToCard(rmapCode);
|
|
|
|
_vm->changeToCard(cardID);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Command 28: disable a movie
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::disableMovie(uint16 op, uint16 argc, uint16 *argv) {
|
2015-07-15 23:59:21 -04:00
|
|
|
VideoHandle handle = _vm->_video->findVideoHandleRiven(argv[0]);
|
|
|
|
if (handle)
|
|
|
|
handle->setEnabled(false);
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Command 29: disable all movies
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::disableAllMovies(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
_vm->_video->disableAllMovies();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Command 31: enable a movie
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::enableMovie(uint16 op, uint16 argc, uint16 *argv) {
|
2015-07-15 23:59:21 -04:00
|
|
|
VideoHandle handle = _vm->_video->findVideoHandleRiven(argv[0]);
|
|
|
|
if (handle)
|
|
|
|
handle->setEnabled(true);
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Command 32: play foreground movie - blocking (movie_id)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::playMovieBlocking(uint16 op, uint16 argc, uint16 *argv) {
|
2011-03-22 20:13:01 -04:00
|
|
|
_vm->_cursor->hideCursor();
|
2011-01-18 20:30:16 +00:00
|
|
|
_vm->_video->playMovieBlockingRiven(argv[0]);
|
2011-03-22 20:13:01 -04:00
|
|
|
_vm->_cursor->showCursor();
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Command 33: play background movie - nonblocking (movie_id)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::playMovie(uint16 op, uint16 argc, uint16 *argv) {
|
2011-01-18 20:30:16 +00:00
|
|
|
_vm->_video->playMovieRiven(argv[0]);
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Command 34: stop a movie
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::stopMovie(uint16 op, uint16 argc, uint16 *argv) {
|
2011-01-18 20:30:16 +00:00
|
|
|
_vm->_video->stopMovieRiven(argv[0]);
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Command 36: unknown
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::unk_36(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
debug(0, "unk_36: Ignoring");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Command 37: fade ambient sounds
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::fadeAmbientSounds(uint16 op, uint16 argc, uint16 *argv) {
|
2011-01-23 07:02:49 +00:00
|
|
|
// Similar to stopSound(), but does fading
|
|
|
|
_vm->_sound->stopAllSLST(true);
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
2011-03-22 20:31:54 -04:00
|
|
|
// Command 38: Store an opcode for use when playing a movie (movie id, time high, time low, opcode, arguments...)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::storeMovieOpcode(uint16 op, uint16 argc, uint16 *argv) {
|
2011-06-14 10:35:58 -04:00
|
|
|
// This opcode is used to delay an opcode's usage based on the elapsed
|
|
|
|
// time of a specified movie. However, every use in the game is for
|
|
|
|
// delaying an activateSLST opcode.
|
|
|
|
|
2011-03-22 20:31:54 -04:00
|
|
|
uint32 scriptSize = 6 + (argc - 4) * 2;
|
2011-01-19 15:22:39 +00:00
|
|
|
|
2011-03-22 20:31:54 -04:00
|
|
|
// Create our dummy script
|
|
|
|
byte *scriptBuf = (byte *)malloc(scriptSize);
|
|
|
|
WRITE_BE_UINT16(scriptBuf, 1); // One command
|
|
|
|
WRITE_BE_UINT16(scriptBuf + 2, argv[3]); // One opcode
|
2011-06-20 00:59:48 +02:00
|
|
|
WRITE_BE_UINT16(scriptBuf + 4, argc - 4); // argc - 4 args
|
2011-03-22 20:31:54 -04:00
|
|
|
|
|
|
|
for (int i = 0; i < argc - 4; i++)
|
|
|
|
WRITE_BE_UINT16(scriptBuf + 6 + (i * 2), argv[i + 4]);
|
|
|
|
|
|
|
|
// Build a script out of 'er
|
|
|
|
Common::SeekableReadStream *scriptStream = new Common::MemoryReadStream(scriptBuf, scriptSize, DisposeAfterUse::YES);
|
2016-08-02 19:43:15 +02:00
|
|
|
RivenScriptPtr script = _vm->_scriptMan->readScript(scriptStream);
|
2011-03-22 20:31:54 -04:00
|
|
|
|
|
|
|
uint32 delayTime = (argv[1] << 16) + argv[2];
|
|
|
|
|
|
|
|
if (delayTime > 0) {
|
|
|
|
// Store the script
|
|
|
|
RivenScriptManager::StoredMovieOpcode storedOp;
|
|
|
|
storedOp.script = script;
|
2011-06-14 10:35:58 -04:00
|
|
|
storedOp.time = delayTime;
|
2011-03-22 20:31:54 -04:00
|
|
|
storedOp.id = argv[0];
|
|
|
|
|
2011-06-14 10:35:58 -04:00
|
|
|
// Store the opcode for later
|
|
|
|
_vm->_scriptMan->setStoredMovieOpcode(storedOp);
|
2011-03-22 20:31:54 -04:00
|
|
|
} else {
|
|
|
|
// Run immediately if we have no delay
|
2016-08-02 21:43:22 +02:00
|
|
|
_vm->_scriptMan->runScript(script, false);
|
2011-03-22 20:31:54 -04:00
|
|
|
}
|
2016-08-02 18:56:55 +02:00
|
|
|
|
|
|
|
delete scriptStream;
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Command 39: activate PLST record (card picture lists)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::activatePLST(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
_vm->_gfx->drawPLST(argv[0]);
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-12-29 23:18:24 +00:00
|
|
|
// An update is automatically sent here as long as it's not a load or update script and updates are enabled.
|
2016-08-02 18:56:55 +02:00
|
|
|
// TODO: Fix the graphics manager
|
|
|
|
//if (_scriptType != kCardLoadScript && _scriptType != kCardUpdateScript)
|
2009-12-29 23:18:24 +00:00
|
|
|
_vm->_gfx->updateScreen();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Command 40: activate SLST record (card ambient sound lists)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::activateSLST(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
// WORKAROUND: Disable the SLST that is played during Riven's intro.
|
|
|
|
// Riven X does this too (spoke this over with Jeff)
|
2014-05-03 22:55:50 -04:00
|
|
|
if (_vm->getCurStack() == kStackTspit && _vm->getCurCardRMAP() == 0x6e9a && argv[0] == 2)
|
2009-12-29 23:18:24 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
_vm->_sound->playSLST(argv[0], _vm->getCurCard());
|
|
|
|
_vm->_activatedSLST = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Command 41: activate MLST record and play
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::activateMLSTAndPlay(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
_vm->_video->activateMLST(argv[0], _vm->getCurCard());
|
2011-01-18 20:30:16 +00:00
|
|
|
_vm->_video->playMovieRiven(argv[0]);
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Command 43: activate BLST record (card hotspot enabling lists)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::activateBLST(uint16 op, uint16 argc, uint16 *argv) {
|
2010-11-20 23:53:14 +00:00
|
|
|
Common::SeekableReadStream* blst = _vm->getResource(ID_BLST, _vm->getCurCard());
|
2009-12-29 23:18:24 +00:00
|
|
|
uint16 recordCount = blst->readUint16BE();
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-12-29 23:18:24 +00:00
|
|
|
for (uint16 i = 0; i < recordCount; i++) {
|
|
|
|
uint16 index = blst->readUint16BE(); // record index
|
|
|
|
uint16 enabled = blst->readUint16BE();
|
|
|
|
uint16 hotspotID = blst->readUint16BE();
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-12-29 23:18:24 +00:00
|
|
|
if (argv[0] == index)
|
|
|
|
for (uint16 j = 0; j < _vm->getHotspotCount(); j++)
|
|
|
|
if (_vm->_hotspots[j].blstID == hotspotID)
|
|
|
|
_vm->_hotspots[j].enabled = (enabled == 1);
|
|
|
|
}
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-12-29 23:18:24 +00:00
|
|
|
delete blst;
|
2011-01-24 02:43:07 +00:00
|
|
|
|
|
|
|
// Recheck our current hotspot because it may have now changed
|
|
|
|
_vm->updateCurrentHotspot();
|
2009-12-29 23:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Command 44: activate FLST record (information on which SFXE resource this card should use)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::activateFLST(uint16 op, uint16 argc, uint16 *argv) {
|
2010-11-20 23:53:14 +00:00
|
|
|
Common::SeekableReadStream* flst = _vm->getResource(ID_FLST, _vm->getCurCard());
|
2009-12-29 23:18:24 +00:00
|
|
|
uint16 recordCount = flst->readUint16BE();
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-12-29 23:18:24 +00:00
|
|
|
for (uint16 i = 0; i < recordCount; i++) {
|
|
|
|
uint16 index = flst->readUint16BE();
|
|
|
|
uint16 sfxeID = flst->readUint16BE();
|
2010-09-08 20:50:56 +00:00
|
|
|
|
|
|
|
if (flst->readUint16BE() != 0)
|
2009-12-29 23:18:24 +00:00
|
|
|
warning("FLST u0 non-zero");
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-12-29 23:18:24 +00:00
|
|
|
if (index == argv[0]) {
|
|
|
|
_vm->_gfx->scheduleWaterEffect(sfxeID);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
delete flst;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Command 45: do zip mode
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::zipMode(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
// Check the ZIPS records to see if we have a match to the hotspot name
|
|
|
|
Common::String hotspotName = _vm->getHotspotName(_vm->getCurHotspot());
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2009-12-29 23:18:24 +00:00
|
|
|
for (uint16 i = 0; i < _vm->_zipModeData.size(); i++)
|
|
|
|
if (_vm->_zipModeData[i].name == hotspotName) {
|
|
|
|
_vm->changeToCard(_vm->_zipModeData[i].id);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Command 46: activate MLST record (movie lists)
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::activateMLST(uint16 op, uint16 argc, uint16 *argv) {
|
2009-12-29 23:18:24 +00:00
|
|
|
_vm->_video->activateMLST(argv[0], _vm->getCurCard());
|
|
|
|
}
|
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::dump(const Common::StringArray &varNames, const Common::StringArray &xNames, byte tabs) {
|
|
|
|
printTabs(tabs);
|
|
|
|
|
|
|
|
if (_type == 7) { // Use the variable name
|
|
|
|
uint16 var = _arguments[0];
|
|
|
|
debugN("%s = %d;\n", varNames[var].c_str(), _arguments[1]);
|
|
|
|
} else if (_type == 17) { // Use the external command name
|
|
|
|
debugN("%s(", xNames[_arguments[0]].c_str());
|
|
|
|
uint16 varCount = _arguments[1];
|
|
|
|
for (uint16 j = 0; j < varCount; j++) {
|
|
|
|
debugN("%d", _arguments[1 + j]);
|
|
|
|
if (j != varCount - 1)
|
|
|
|
debugN(", ");
|
|
|
|
}
|
|
|
|
debugN(");\n");
|
|
|
|
} else if (_type == 24) { // Use the variable name
|
|
|
|
uint16 var = _arguments[0];
|
|
|
|
debugN("%s += %d;\n", varNames[var].c_str(), _arguments[1]);
|
|
|
|
} else {
|
|
|
|
debugN("%s(", _opcodes[_type].desc);
|
|
|
|
for (uint16 j = 0; j < _arguments.size(); j++) {
|
|
|
|
debugN("%d", _arguments[j]);
|
|
|
|
if (j != _arguments.size() - 1)
|
|
|
|
debugN(", ");
|
|
|
|
}
|
|
|
|
debugN(");\n");
|
|
|
|
}
|
2010-07-09 16:53:20 +00:00
|
|
|
}
|
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSimpleCommand::execute() {
|
|
|
|
uint16 *argValues = new uint16[_arguments.size()];
|
2011-03-22 20:31:54 -04:00
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
for (uint16 k = 0; k < _arguments.size(); k++)
|
|
|
|
argValues[k] = _arguments[k];
|
|
|
|
|
|
|
|
debug (4, "Running opcode %04x, argument count %d", _type, _arguments.size());
|
|
|
|
(this->*(_opcodes[_type].proc)) (_type, _arguments.size(), argValues);
|
|
|
|
|
|
|
|
delete[] argValues;
|
2010-07-09 16:53:20 +00:00
|
|
|
}
|
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
RivenSwitchCommand::RivenSwitchCommand(MohawkEngine_Riven *vm) :
|
|
|
|
RivenCommand(vm),
|
|
|
|
_variableId(0) {
|
2010-07-09 16:53:20 +00:00
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
}
|
2010-07-09 16:53:20 +00:00
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
RivenSwitchCommand::~RivenSwitchCommand() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
RivenSwitchCommand *RivenSwitchCommand::createFromStream(MohawkEngine_Riven *vm, int type, Common::ReadStream *stream) {
|
|
|
|
RivenSwitchCommand *command = new RivenSwitchCommand(vm);
|
2010-07-09 16:53:20 +00:00
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
if (stream->readUint16BE() != 2) {
|
|
|
|
// This value is not used in the original engine
|
|
|
|
warning("if-then-else unknown value is not 2");
|
2010-07-09 16:53:20 +00:00
|
|
|
}
|
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
// variable to check against
|
|
|
|
command->_variableId = stream->readUint16BE();
|
2010-07-09 16:53:20 +00:00
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
// number of logic blocks
|
|
|
|
uint16 logicBlockCount = stream->readUint16BE();
|
|
|
|
command->_branches.resize(logicBlockCount);
|
2010-09-01 13:28:12 +00:00
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
for (uint16 i = 0; i < logicBlockCount; i++) {
|
|
|
|
Branch &branch = command->_branches[i];
|
|
|
|
|
|
|
|
// Value for this logic block
|
|
|
|
branch.value = stream->readUint16BE();
|
2016-08-02 19:43:15 +02:00
|
|
|
branch.script = vm->_scriptMan->readScript(stream);
|
2010-07-09 16:53:20 +00:00
|
|
|
}
|
2009-12-29 23:18:24 +00:00
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
return command;
|
2011-03-22 20:31:54 -04:00
|
|
|
}
|
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSwitchCommand::dump(const Common::StringArray &varNames, const Common::StringArray &xNames, byte tabs) {
|
|
|
|
printTabs(tabs); debugN("switch (%s) {\n", varNames[_variableId].c_str());
|
|
|
|
for (uint16 j = 0; j < _branches.size(); j++) {
|
|
|
|
printTabs(tabs + 1);
|
|
|
|
if (_branches[j].value == 0xFFFF)
|
|
|
|
debugN("default:\n");
|
|
|
|
else
|
|
|
|
debugN("case %d:\n", _branches[j].value);
|
|
|
|
_branches[j].script->dumpScript(varNames, xNames, tabs + 2);
|
|
|
|
printTabs(tabs + 2); debugN("break;\n");
|
2011-03-22 20:31:54 -04:00
|
|
|
}
|
2016-08-02 18:56:55 +02:00
|
|
|
printTabs(tabs); debugN("}\n");
|
2011-03-22 20:31:54 -04:00
|
|
|
}
|
|
|
|
|
2016-08-02 18:56:55 +02:00
|
|
|
void RivenSwitchCommand::execute() {
|
|
|
|
// Get the switch variable value
|
|
|
|
uint32 value = _vm->getStackVar(_variableId);
|
|
|
|
|
|
|
|
// Look for a case matching the value
|
|
|
|
for (uint i = 0; i < _branches.size(); i++) {
|
|
|
|
if (_branches[i].value == value) {
|
2016-08-02 21:43:22 +02:00
|
|
|
_vm->_scriptMan->runScript(_branches[i].script, false);
|
2016-08-02 18:56:55 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Look for the default case if any
|
|
|
|
for (uint i = 0; i < _branches.size(); i++) {
|
|
|
|
if (_branches[i].value == 0Xffff) {
|
2016-08-02 21:43:22 +02:00
|
|
|
_vm->_scriptMan->runScript(_branches[i].script, false);
|
2016-08-02 18:56:55 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2011-03-22 20:31:54 -04:00
|
|
|
}
|
|
|
|
|
2009-12-29 23:18:24 +00:00
|
|
|
} // End of namespace Mohawk
|