scummvm/engines/twine/script/script_life_v1.cpp
2020-12-29 00:16:46 +01:00

1909 lines
59 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 2
* 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, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
*/
#include "twine/script/script_life_v1.h"
#include "common/memstream.h"
#include "common/stream.h"
#include "twine/scene/actor.h"
#include "twine/scene/animations.h"
#include "twine/audio/music.h"
#include "twine/audio/sound.h"
#include "twine/scene/collision.h"
#include "twine/flamovies.h"
#include "twine/scene/gamestate.h"
#include "twine/scene/grid.h"
#include "twine/holomap.h"
#include "twine/input.h"
#include "twine/menu/interface.h"
#include "twine/menu/menu.h"
#include "twine/scene/movements.h"
#include "twine/renderer/redraw.h"
#include "twine/renderer/renderer.h"
#include "twine/renderer/screens.h"
#include "twine/resources/resources.h"
#include "twine/scene/scene.h"
#include "twine/text.h"
#include "twine/twine.h"
namespace TwinE {
// TODO: validate that lTEXTCLEAR is called before lTEXT - if that is the case
// move this into the LifeScriptContext
static int32 drawVar1;
struct LifeScriptContext {
int32 actorIdx;
ActorStruct *actor;
Common::MemorySeekableReadWriteStream stream;
uint8 *opcodePtr; // local opcode script pointer
LifeScriptContext(int32 _actorIdx, ActorStruct *_actor) : actorIdx(_actorIdx), actor(_actor), stream(_actor->lifeScript, _actor->lifeScriptSize) {
assert(actor->positionInLifeScript >= 0);
stream.skip(_actor->positionInLifeScript);
updateOpcodePos();
}
void setOpcode(uint8 opcode) {
*opcodePtr = opcode;
}
void updateOpcodePos() {
opcodePtr = actor->lifeScript + stream.pos();
}
};
/**
* Returns @c -1 Need implementation, @c 0 Condition false, @c 1 - Condition true
*/
typedef int32 ScriptLifeFunc(TwinEEngine *engine, LifeScriptContext &ctx);
struct ScriptLifeFunction {
const char *name;
ScriptLifeFunc *function;
};
#define MAPFUNC(name, func) \
{ name, func }
/** Script condition operators */
enum LifeScriptOperators {
/*==*/kEqualTo = 0,
/*> */ kGreaterThan = 1,
/*< */ kLessThan = 2,
/*>=*/kGreaterThanOrEqualTo = 3,
/*<=*/kLessThanOrEqualTo = 4,
/*!=*/kNotEqualTo = 5
};
/** Script condition command opcodes */
enum LifeScriptConditions {
/*0x00*/ kcCOL = 0, /*<! Current actor collision with another actor. (Parameter = Actor Index) */
/*0x01*/ kcCOL_OBJ = 1, /*<! Actor collision with the actor passed as parameter. (Parameter = Actor Index, Parameter = Actor Index) */
/*0x02*/ kcDISTANCE = 2, /*<! Distance between the current actor and the actor passed as parameter. (Parameter = Actor Index, Parameter = Distance between) */
/*0x03*/ kcZONE = 3, /*<! Current actor tread on zone passed as parameter. (Parameter = Zone Index) */
/*0x04*/ kcZONE_OBJ = 4, /*<! The actor passed as parameter will tread on zone passed as parameter. (Parameter = Actor Index, Parameter = Zone Index) */
/*0x05*/ kcBODY = 5, /*<! Body of the current actor. (Parameter = Body Index) */
/*0x06*/ kcBODY_OBJ = 6, /*<! Body of the actor passed as parameter. (Parameter = Body Index) */
/*0x07*/ kcANIM = 7, /*<! Body Animation of the current actor. (Parameter = Animation Index) */
/*0x08*/ kcANIM_OBJ = 8, /*<! Body Animation of the actor passed as parameter. (Parameter = Animation Index) */
/*0x09*/ kcL_TRACK = 9, /*<! Current actor track. (Parameter = Track Index) */
/*0x0A*/ kcL_TRACK_OBJ = 10, /*<! Track of the actor passed as parameter. (Parameter = Track Index) */
/*0x0B*/ kcFLAG_CUBE = 11, /*<! Game Cube Flags. (Parameter = Cube Flag Index, Parameter = 0 (not set), = 1 (set))k */
/*0x0C*/ kcCONE_VIEW = 12, /*<! The actor passed as parameter have a "vision in circle". (Parameter = Actor Index, Parameter = Distance) */
/*0x0D*/ kcHIT_BY = 13, /*<! Current actor hited by the actor passed as parameter. (Parameter = Actor Index) */
/*0x0E*/ kcACTION = 14, /*<! Hero action behavior. (Parameter = Behaviour Index) */
/*0x0F*/ kcFLAG_GAME = 15, /*<! Game Flags (See further list). (Parameter = Flag Index, Parameter = 0 (not set), = 1 (set)) */
/*0x10*/ kcLIFE_POINT = 16, /*<! Current actor life points. (Parameter = Life points) */
/*0x11*/ kcLIFE_POINT_OBJ = 17, /*<! Life points of the current actor passed as parameter. (Parameter = Life points) */
/*0x12*/ kcNUM_LITTLE_KEYS = 18, /*<! Number of keys. (Parameter = Number of keys) */
/*0x13*/ kcNUM_GOLD_PIECES = 19, /*<! Coins/Gold Amount. (Parameter = Coins/Gold amount) */
/*0x14*/ kcBEHAVIOUR = 20, /*<! Hero behaviour. (Parameter = Behaviour Index) */
/*0x15*/ kcCHAPTER = 21, /*<! Story Chapters. (Parameter = Chapter Index) */
/*0x16*/ kcDISTANCE_3D = 22, /*<! Distance between the actor passed as parameter and the current actor. (Parameter = Actor Index, Parameter = Distance) */
/*0x17 - 23 unused */
/*0x18 - 24 unused */
/*0x19*/ kcUSE_INVENTORY = 25, /*<! Use inventory object. (Parameter = Object Index in the inventory, Paramenter = 0 (Not in Inventory), = 1 (In the Inventory)) */
/*0x1A*/ kcCHOICE = 26, /*<! Menu choice. (Parameter = Text Index in the current Text Bank) */
/*0x1B*/ kcFUEL = 27, /*<! Amount of fuel gas the Hero have in his inventory. (Parameter = Gas amount) */
/*0x1C*/ kcCARRIED_BY = 28, /*<! The current is carried by the actor passed as paramenter. (Parameter = Actor Index) */
/*0x1D*/ kcCDROM = 29 /*<! CDROM audio tracks. (Parameter = Audio Tracks Index) */
};
/**
* Returns @c 1 Condition value size (1 byte), @c 2 Condition value size (2 byes)
*/
static int32 processLifeConditions(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 conditionValueSize = 1;
int32 conditionOpcode = ctx.stream.readByte();
switch (conditionOpcode) {
case kcCOL:
if (ctx.actor->life <= 0) {
engine->_scene->currentScriptValue = -1;
} else {
engine->_scene->currentScriptValue = ctx.actor->collision;
}
break;
case kcCOL_OBJ: {
int32 actorIdx = ctx.stream.readByte();
if (engine->_scene->getActor(actorIdx)->life <= 0) {
engine->_scene->currentScriptValue = -1;
} else {
engine->_scene->currentScriptValue = engine->_scene->getActor(actorIdx)->collision;
}
break;
}
case kcDISTANCE: {
int32 actorIdx = ctx.stream.readByte();
conditionValueSize = 2;
ActorStruct *otherActor = engine->_scene->getActor(actorIdx);
if (!otherActor->dynamicFlags.bIsDead) {
if (ABS(ctx.actor->y - otherActor->y) >= 1500) {
engine->_scene->currentScriptValue = MAX_TARGET_ACTOR_DISTANCE;
} else {
// Returns int32, so we check for integer overflow
int32 distance = engine->_movements->getDistance2D(ctx.actor->x, ctx.actor->z, otherActor->x, otherActor->z);
if (ABS(distance) > MAX_TARGET_ACTOR_DISTANCE) {
engine->_scene->currentScriptValue = MAX_TARGET_ACTOR_DISTANCE;
} else {
engine->_scene->currentScriptValue = distance;
}
}
} else {
engine->_scene->currentScriptValue = MAX_TARGET_ACTOR_DISTANCE;
}
break;
}
case kcZONE:
engine->_scene->currentScriptValue = ctx.actor->zone;
break;
case kcZONE_OBJ: {
int32 actorIdx = ctx.stream.readByte();
engine->_scene->currentScriptValue = engine->_scene->getActor(actorIdx)->zone;
break;
}
case kcBODY:
engine->_scene->currentScriptValue = ctx.actor->body;
break;
case kcBODY_OBJ: {
int32 actorIdx = ctx.stream.readByte();
engine->_scene->currentScriptValue = engine->_scene->getActor(actorIdx)->body;
break;
}
case kcANIM:
engine->_scene->currentScriptValue = (int16)ctx.actor->anim;
break;
case kcANIM_OBJ: {
int32 actorIdx = ctx.stream.readByte();
engine->_scene->currentScriptValue = (int16)engine->_scene->getActor(actorIdx)->anim;
break;
}
case kcL_TRACK:
engine->_scene->currentScriptValue = ctx.actor->labelIdx;
break;
case kcL_TRACK_OBJ: {
int32 actorIdx = ctx.stream.readByte();
engine->_scene->currentScriptValue = engine->_scene->getActor(actorIdx)->labelIdx;
break;
}
case kcFLAG_CUBE: {
int32 flagIdx = ctx.stream.readByte();
engine->_scene->currentScriptValue = engine->_scene->sceneFlags[flagIdx];
break;
}
case kcCONE_VIEW: {
int32 newAngle = 0;
int32 targetActorIdx = ctx.stream.readByte();
ActorStruct *targetActor = engine->_scene->getActor(targetActorIdx);
conditionValueSize = 2;
if (!targetActor->dynamicFlags.bIsDead) {
if (ABS(targetActor->y - ctx.actor->y) < 1500) {
newAngle = engine->_movements->getAngleAndSetTargetActorDistance(ctx.actor->x, ctx.actor->z, targetActor->x, targetActor->z);
if (ABS(engine->_movements->targetActorDistance) > MAX_TARGET_ACTOR_DISTANCE) {
engine->_movements->targetActorDistance = MAX_TARGET_ACTOR_DISTANCE;
}
} else {
engine->_movements->targetActorDistance = MAX_TARGET_ACTOR_DISTANCE;
}
if (IS_HERO(targetActorIdx)) {
int32 heroAngle = ClampAngle(ctx.actor->angle + ANGLE_360 + ANGLE_45 - newAngle + ANGLE_360);
if (ABS(heroAngle) > ANGLE_90) {
engine->_scene->currentScriptValue = MAX_TARGET_ACTOR_DISTANCE;
} else {
engine->_scene->currentScriptValue = engine->_movements->targetActorDistance;
}
} else {
if (engine->_actor->heroBehaviour == HeroBehaviourType::kDiscrete) {
int32 heroAngle = ClampAngle(ctx.actor->angle + ANGLE_360 + ANGLE_45 - newAngle + ANGLE_360);
if (ABS(heroAngle) > ANGLE_90) {
engine->_scene->currentScriptValue = MAX_TARGET_ACTOR_DISTANCE;
} else {
engine->_scene->currentScriptValue = engine->_movements->targetActorDistance;
}
} else {
engine->_scene->currentScriptValue = engine->_movements->targetActorDistance;
}
}
} else {
engine->_scene->currentScriptValue = MAX_TARGET_ACTOR_DISTANCE;
}
break;
}
case kcHIT_BY:
engine->_scene->currentScriptValue = ctx.actor->hitBy;
break;
case kcACTION:
engine->_scene->currentScriptValue = engine->_movements->shouldTriggerZoneAction() ? 1 : 0;
break;
case kcFLAG_GAME: {
int32 flagIdx = ctx.stream.readByte();
if (!engine->_gameState->inventoryDisabled() ||
(engine->_gameState->inventoryDisabled() && flagIdx >= MaxInventoryItems)) {
engine->_scene->currentScriptValue = engine->_gameState->gameFlags[flagIdx];
} else {
if (flagIdx == GAMEFLAG_INVENTORY_DISABLED) {
engine->_scene->currentScriptValue = engine->_gameState->inventoryDisabled();
} else {
engine->_scene->currentScriptValue = 0;
}
}
break;
}
case kcLIFE_POINT:
engine->_scene->currentScriptValue = ctx.actor->life;
break;
case kcLIFE_POINT_OBJ: {
int32 actorIdx = ctx.stream.readByte();
engine->_scene->currentScriptValue = engine->_scene->getActor(actorIdx)->life;
break;
}
case kcNUM_LITTLE_KEYS:
engine->_scene->currentScriptValue = engine->_gameState->inventoryNumKeys;
break;
case kcNUM_GOLD_PIECES:
conditionValueSize = 2;
engine->_scene->currentScriptValue = engine->_gameState->inventoryNumKashes;
break;
case kcBEHAVIOUR:
engine->_scene->currentScriptValue = (int16)engine->_actor->heroBehaviour;
break;
case kcCHAPTER:
engine->_scene->currentScriptValue = engine->_gameState->gameChapter;
break;
case kcDISTANCE_3D: {
int32 targetActorIdx;
ActorStruct *targetActor;
targetActorIdx = ctx.stream.readByte();
targetActor = engine->_scene->getActor(targetActorIdx);
conditionValueSize = 2;
if (!targetActor->dynamicFlags.bIsDead) {
// Returns int32, so we check for integer overflow
int32 distance = engine->_movements->getDistance3D(ctx.actor->x, ctx.actor->y, ctx.actor->z, targetActor->x, targetActor->y, targetActor->z);
if (ABS(distance) > MAX_TARGET_ACTOR_DISTANCE) {
engine->_scene->currentScriptValue = MAX_TARGET_ACTOR_DISTANCE;
} else {
engine->_scene->currentScriptValue = distance;
}
} else {
engine->_scene->currentScriptValue = MAX_TARGET_ACTOR_DISTANCE;
}
break;
}
case 23: // unused
case 24:
break;
case kcUSE_INVENTORY: {
int32 item = ctx.stream.readByte();
if (!engine->_gameState->inventoryDisabled()) {
if (item == engine->loopInventoryItem) {
engine->_scene->currentScriptValue = 1;
} else {
if (engine->_gameState->inventoryFlags[item] == 1 && engine->_gameState->hasItem((InventoryItems)item)) {
engine->_scene->currentScriptValue = 1;
} else {
engine->_scene->currentScriptValue = 0;
}
}
if (engine->_scene->currentScriptValue == 1) {
engine->_redraw->addOverlay(OverlayType::koInventoryItem, item, 0, 0, 0, OverlayPosType::koNormal, 3);
}
} else {
engine->_scene->currentScriptValue = 0;
}
break;
}
case kcCHOICE:
conditionValueSize = 2;
engine->_scene->currentScriptValue = engine->_gameState->choiceAnswer;
break;
case kcFUEL:
engine->_scene->currentScriptValue = engine->_gameState->inventoryNumGas;
break;
case kcCARRIED_BY:
engine->_scene->currentScriptValue = ctx.actor->standOn;
break;
case kcCDROM:
engine->_scene->currentScriptValue = 1;
break;
default:
error("Actor condition opcode %d", conditionOpcode);
break;
}
return conditionValueSize;
}
/**
* Returns @c -1 Need implementation, @c 0 Condition false, @c 1 Condition true
*/
static int32 processLifeOperators(TwinEEngine *engine, LifeScriptContext &ctx, int32 valueSize) {
const int32 operatorCode = ctx.stream.readByte();
int32 conditionValue;
if (valueSize == 1) {
conditionValue = ctx.stream.readByte();
} else if (valueSize == 2) {
conditionValue = ctx.stream.readSint16LE();
} else {
error("Unknown operator value size %d", valueSize);
}
switch (operatorCode) {
case kEqualTo:
if (engine->_scene->currentScriptValue == conditionValue) {
return 1;
}
break;
case kGreaterThan:
if (engine->_scene->currentScriptValue > conditionValue) {
return 1;
}
break;
case kLessThan:
if (engine->_scene->currentScriptValue < conditionValue) {
return 1;
}
break;
case kGreaterThanOrEqualTo:
if (engine->_scene->currentScriptValue >= conditionValue) {
return 1;
}
break;
case kLessThanOrEqualTo:
if (engine->_scene->currentScriptValue <= conditionValue) {
return 1;
}
break;
case kNotEqualTo:
if (engine->_scene->currentScriptValue != conditionValue) {
return 1;
}
break;
default:
warning("Unknown life script operator opcode %d", operatorCode);
break;
}
return 0;
}
/** Life script command definitions */
/**
* For unused opcodes
*/
static int32 lEMPTY(TwinEEngine *engine, LifeScriptContext &ctx) {
return 0;
}
/**
* End of Actor Life Script
* @note Opcode @c 0x00
*/
static int32 lEND(TwinEEngine *engine, LifeScriptContext &ctx) {
ctx.actor->positionInLifeScript = -1;
return 1; // break script
}
/**
* No Operation
* @note Opcode @c 0x01
*/
static int32 lNOP(TwinEEngine *engine, LifeScriptContext &ctx) {
ctx.stream.skip(1);
return 0;
}
/**
* To execute a switch no if. It's used to toggle the switch.
* @note Opcode @c 0x02
*/
static int32 lSNIF(TwinEEngine *engine, LifeScriptContext &ctx) {
const int32 valueSize = processLifeConditions(engine, ctx);
if (!processLifeOperators(engine, ctx, valueSize)) {
ctx.setOpcode(0x0D); // SWIF
}
ctx.stream.seek(ctx.stream.readSint16LE()); // condition offset
return 0;
}
/**
* To jump to another offset in the current script. (Parameter = Offset)
* @note Opcode @c 0x03
*/
static int32 lOFFSET(TwinEEngine *engine, LifeScriptContext &ctx) {
ctx.stream.seek(ctx.stream.readSint16LE()); // offset
return 0;
}
/**
* Will never execute that condition.
* @note Opcode @c 0x04
*/
static int32 lNEVERIF(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 valueSize = processLifeConditions(engine, ctx);
processLifeOperators(engine, ctx, valueSize);
ctx.stream.seek(ctx.stream.readSint16LE()); // condition offset
return 0;
}
/**
* Will not execute the condition.
* @note Opcode @c 0x06
*/
static int32 lNO_IF(TwinEEngine *engine, LifeScriptContext &ctx) {
return 0;
}
/**
* Specify a new label
* @note Opcode @c 0x0A
*/
static int32 lLABEL(TwinEEngine *engine, LifeScriptContext &ctx) {
ctx.stream.skip(1); // label id - script offset
return 0;
}
/**
* To stop running the current script
* @note Opcode @c 0x0B
*/
static int32 lRETURN(TwinEEngine *engine, LifeScriptContext &ctx) {
return 1; // break script
}
/**
* Do a certain statement according the condition.
* @note Opcode @c 0x0C
*/
static int32 lIF(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 valueSize = processLifeConditions(engine, ctx);
if (!processLifeOperators(engine, ctx, valueSize)) {
ctx.stream.seek(ctx.stream.readSint16LE()); // condition offset
} else {
ctx.stream.skip(2);
}
return 0;
}
/**
* To execute a switch if.
* @note Opcode @c 0x0D
*/
static int32 lSWIF(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 valueSize = processLifeConditions(engine, ctx);
if (!processLifeOperators(engine, ctx, valueSize)) {
ctx.stream.seek(ctx.stream.readSint16LE()); // condition offset
} else {
ctx.stream.skip(2);
ctx.setOpcode(0x02); // SNIF
}
return 0;
}
/**
* Will only execute that condition one time.
* @note Opcode @c 0x0E
*/
static int32 lONEIF(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 valueSize = processLifeConditions(engine, ctx);
if (!processLifeOperators(engine, ctx, valueSize)) {
ctx.stream.seek(ctx.stream.readSint16LE()); // condition offset
} else {
ctx.stream.skip(2);
ctx.setOpcode(0x04); // NEVERIF
}
return 0;
}
/**
* Else statement for an IF condition.
* @note Opcode @c 0x0F
*/
static int32 lELSE(TwinEEngine *engine, LifeScriptContext &ctx) {
ctx.stream.seek(ctx.stream.readSint16LE()); // offset
return 0;
}
/**
* Choose new body for the current actor (Parameter = File3D Body Instance)
* @note Opcode @c 0x11
*/
static int32 lBODY(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 bodyIdx = ctx.stream.readByte();
engine->_actor->initModelActor(bodyIdx, ctx.actorIdx);
return 0;
}
/**
* Choose new body for the actor passed as parameter (Parameter = Actor Index, Parameter = File3D Body Instance)
* @note Opcode @c 0x12
*/
static int32 lBODY_OBJ(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 otherActorIdx = ctx.stream.readByte();
int32 otherBodyIdx = ctx.stream.readByte();
engine->_actor->initModelActor(otherBodyIdx, otherActorIdx);
return 0;
}
/**
* Choose new animation for the current actor (Parameter = File3D Animation Instance)
* @note Opcode @c 0x13
*/
static int32 lANIM(TwinEEngine *engine, LifeScriptContext &ctx) {
AnimationTypes animIdx = (AnimationTypes)ctx.stream.readByte();
engine->_animations->initAnim(animIdx, 0, AnimationTypes::kStanding, ctx.actorIdx);
return 0;
}
/**
* Choose new animation for the actor passed as parameter (Parameter = Actor Index, Parameter = File3D Animation Instance)
* @note Opcode @c 0x14
*/
static int32 lANIM_OBJ(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 otherActorIdx = ctx.stream.readByte();
AnimationTypes otherAnimIdx = (AnimationTypes)ctx.stream.readByte();
engine->_animations->initAnim(otherAnimIdx, 0, AnimationTypes::kStanding, otherActorIdx);
return 0;
}
/**
* Same as SET_COMPORTAMENT
* @note Opcode @c 0x15
*/
static int32 lSET_LIFE(TwinEEngine *engine, LifeScriptContext &ctx) {
ctx.actor->positionInLifeScript = ctx.stream.readSint16LE(); // offset
return 0;
}
/**
* Same as SET_COMPORTAMENT_OBJ
* @note Opcode @c 0x16
*/
static int32 lSET_LIFE_OBJ(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 otherActorIdx = ctx.stream.readByte();
engine->_scene->getActor(otherActorIdx)->positionInLifeScript = ctx.stream.readSint16LE(); // offset
return 0;
}
/**
* Set a new track for the current actor. (Parameter = Track offset)
* @note Opcode @c 0x17
*/
static int32 lSET_TRACK(TwinEEngine *engine, LifeScriptContext &ctx) {
ctx.actor->positionInMoveScript = ctx.stream.readSint16LE(); // offset
return 0;
}
/**
* Set a new track for tha actor passed as parameter (Parameter = Actor Index, Parameter = Track offset)
* @note Opcode @c 0x18
*/
static int32 lSET_TRACK_OBJ(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 otherActorIdx = ctx.stream.readByte();
engine->_scene->getActor(otherActorIdx)->positionInMoveScript = ctx.stream.readSint16LE(); // offset
return 0;
}
/**
* Choose a message to display. (Parameter = Text Index in the current Text Bank)
* @note Opcode @c 0x19
*/
static int32 lMESSAGE(TwinEEngine *engine, LifeScriptContext &ctx) {
const int32 textIdx = ctx.stream.readSint16LE();
engine->freezeTime();
if (engine->_text->showDialogueBubble) {
engine->_redraw->drawBubble(ctx.actorIdx);
}
engine->_text->setFontCrossColor(ctx.actor->talkColor);
engine->_scene->talkingActor = ctx.actorIdx;
engine->_text->drawTextFullscreen(textIdx);
engine->unfreezeTime();
engine->_redraw->redrawEngineActions(true);
return 0;
}
/**
* To set the current actor static flag fallable. (Parameter = value & 1)
* @note Opcode @c 0x1A
*/
static int32 lFALLABLE(TwinEEngine *engine, LifeScriptContext &ctx) {
const int32 flag = ctx.stream.readByte();
ctx.actor->staticFlags.bCanFall = flag & 1;
return 0;
}
/**
* To set direction for current actor.
* @note Opcode @c 0x1B
*/
static int32 lSET_DIRMODE(TwinEEngine *engine, LifeScriptContext &ctx) {
const int32 controlMode = ctx.stream.readByte();
ctx.actor->controlMode = (ControlMode)controlMode;
if (ctx.actor->controlMode == ControlMode::kFollow || ctx.actor->controlMode == ControlMode::kFollow2) {
ctx.actor->followedActor = ctx.stream.readByte();
}
return 0;
}
/**
* To set direction
* @note Opcode @c 0x1C
*/
static int32 lSET_DIRMODE_OBJ(TwinEEngine *engine, LifeScriptContext &ctx) {
const int32 otherActorIdx = ctx.stream.readByte();
const int32 controlMode = ctx.stream.readByte();
ActorStruct *otherActor = engine->_scene->getActor(otherActorIdx);
otherActor->controlMode = (ControlMode)controlMode;
if (otherActor->controlMode == ControlMode::kFollow || ctx.actor->controlMode == ControlMode::kFollow2) {
otherActor->followedActor = ctx.stream.readByte();
}
return 0;
}
/**
* Camara follow the actor (Parameter = Actor to Follow)
* @note Opcode @c 0x1D
*/
static int32 lCAM_FOLLOW(TwinEEngine *engine, LifeScriptContext &ctx) {
const int32 followedActorIdx = ctx.stream.readByte();
if (engine->_scene->currentlyFollowedActor != followedActorIdx) {
const ActorStruct *followedActor = engine->_scene->getActor(followedActorIdx);
engine->_grid->newCameraX = followedActor->x / BRICK_SIZE;
engine->_grid->newCameraY = followedActor->y / BRICK_HEIGHT;
engine->_grid->newCameraZ = followedActor->z / BRICK_SIZE;
engine->_scene->currentlyFollowedActor = followedActorIdx;
engine->_redraw->reqBgRedraw = true;
}
return 0;
}
/**
* Set a new behavior for Twinsen (Paramenter = Behavior Index)
* @note Opcode @c 0x1E
*/
static int32 lSET_BEHAVIOUR(TwinEEngine *engine, LifeScriptContext &ctx) {
const HeroBehaviourType behavior = (HeroBehaviourType)ctx.stream.readByte();
engine->_animations->initAnim(AnimationTypes::kStanding, 0, AnimationTypes::kAnimInvalid, 0);
engine->_actor->setBehaviour(behavior);
return 0;
}
/**
* Set a new value for the cube flag (Paramter = Cube Flag Index, Parameter = Value)
* @note Opcode @c 0x1F
*/
static int32 lSET_FLAG_CUBE(TwinEEngine *engine, LifeScriptContext &ctx) {
const int32 flagIdx = ctx.stream.readByte();
const int32 flagValue = ctx.stream.readByte();
engine->_scene->sceneFlags[flagIdx] = flagValue;
return 0;
}
/**
* Set a new behaviour for the current actor. (Paramter = Comportament number)
* @note Opcode @c 0x20
*/
static int32 lCOMPORTEMENT(TwinEEngine *engine, LifeScriptContext &ctx) {
ctx.stream.skip(1);
return 0;
}
/**
* Set a new comportament for the current actor. (Parameter = Comportament Offset)
* @note Opcode @c 0x21
*/
static int32 lSET_COMPORTEMENT(TwinEEngine *engine, LifeScriptContext &ctx) {
ctx.actor->positionInLifeScript = ctx.stream.readSint16LE();
return 0;
}
/**
* Set a new comportament for the actor passed as parameter. (Paramter = Actor Index, Parameter = Comportament Offset)
* @note Opcode @c 0x22
*/
static int32 lSET_COMPORTEMENT_OBJ(TwinEEngine *engine, LifeScriptContext &ctx) {
const int32 otherActorIdx = ctx.stream.readByte();
engine->_scene->getActor(otherActorIdx)->positionInLifeScript = ctx.stream.readSint16LE();
return 0;
}
/**
* End of comportament.
* @note Opcode @c 0x23
*/
static int32 lEND_COMPORTEMENT(TwinEEngine *engine, LifeScriptContext &ctx) {
return 1; // break
}
/**
* Set a new value for the game flag (Paramter = Game Flag Index, Parameter = Value)
* @note Opcode @c 0x24
*/
static int32 lSET_FLAG_GAME(TwinEEngine *engine, LifeScriptContext &ctx) {
const uint8 flagIdx = ctx.stream.readByte();
const uint8 flagValue = ctx.stream.readByte();
engine->_gameState->setFlag(flagIdx, flagValue);
return 0;
}
/**
* Kill the actor passed as paramenter (Parameter = Actor Index)
* @note Opcode @c 0x25
*/
static int32 lKILL_OBJ(TwinEEngine *engine, LifeScriptContext &ctx) {
const int32 otherActorIdx = ctx.stream.readByte();
engine->_actor->processActorCarrier(otherActorIdx);
ActorStruct *otherActor = engine->_scene->getActor(otherActorIdx);
otherActor->dynamicFlags.bIsDead = 1;
otherActor->entity = -1;
otherActor->zone = -1;
otherActor->life = 0;
return 0;
}
/**
* Kill the current actor
* @note Opcode @c 0x26
*/
static int32 lSUICIDE(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_actor->processActorCarrier(ctx.actorIdx);
ctx.actor->dynamicFlags.bIsDead = 1;
ctx.actor->entity = -1;
ctx.actor->zone = -1;
ctx.actor->life = 0;
return 0;
}
/**
* Use one key collected in the behaviors menu.
* @note Opcode @c 0x27
*/
static int32 lUSE_ONE_LITTLE_KEY(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_gameState->inventoryNumKeys--;
if (engine->_gameState->inventoryNumKeys < 0) {
engine->_gameState->inventoryNumKeys = 0;
}
engine->_redraw->addOverlay(OverlayType::koSprite, SPRITEHQR_KEY, 0, 0, 0, OverlayPosType::koFollowActor, 1);
return 0;
}
/**
* To give money. (Paramenter = Amount)
* @note Opcode @c 0x28
*/
static int32 lGIVE_GOLD_PIECES(TwinEEngine *engine, LifeScriptContext &ctx) {
int16 oldNumKashes = engine->_gameState->inventoryNumKashes;
bool hideRange = false;
int16 kashes = ctx.stream.readSint16LE();
engine->_gameState->inventoryNumKashes -= kashes;
if (engine->_gameState->inventoryNumKashes < 0) {
engine->_gameState->inventoryNumKashes = 0;
}
engine->_redraw->addOverlay(OverlayType::koSprite, SPRITEHQR_KASHES, 10, 15, 0, OverlayPosType::koNormal, 3);
for (int16 i = 0; i < OVERLAY_MAX_ENTRIES; i++) {
OverlayListStruct *overlay = &engine->_redraw->overlayList[i];
if (overlay->info0 != -1 && overlay->type == OverlayType::koNumberRange) {
overlay->info0 = engine->_collision->getAverageValue(overlay->info1, overlay->info0, 100, overlay->lifeTime - engine->lbaTime - 50);
overlay->info1 = engine->_gameState->inventoryNumKashes;
overlay->lifeTime = engine->lbaTime + 150;
hideRange = true;
break;
}
}
if (!hideRange) {
engine->_redraw->addOverlay(OverlayType::koNumberRange, oldNumKashes, 50, 20, engine->_gameState->inventoryNumKashes, OverlayPosType::koNormal, 3);
}
return 0;
}
/**
* The game will not play the current actor script anymore
* @note Opcode @c 0x29
*/
static int32 lEND_LIFE(TwinEEngine *engine, LifeScriptContext &ctx) {
ctx.actor->positionInLifeScript = -1;
return 1; // break;
}
/**
* The current actor will stop doing the track.
* @note Opcode @c 0x2A
*/
static int32 lSTOP_L_TRACK(TwinEEngine *engine, LifeScriptContext &ctx) {
ctx.actor->pausedTrackPtr = ctx.actor->currentLabelPtr;
ctx.actor->positionInMoveScript = -1;
return 0;
}
/**
* The current actor will resume the tracked started before.
* @note Opcode @c 0x2B
*/
static int32 lRESTORE_L_TRACK(TwinEEngine *engine, LifeScriptContext &ctx) {
ctx.actor->positionInMoveScript = ctx.actor->pausedTrackPtr;
return 0;
}
/**
* The actor passed as parameter will say that massage (Parameter = Actor Index, Parameter = Text Index in the current Text Bank)
* @note Opcode @c 0x2C
*/
static int32 lMESSAGE_OBJ(TwinEEngine *engine, LifeScriptContext &ctx) {
const int32 otherActorIdx = ctx.stream.readByte();
const int32 textIdx = ctx.stream.readSint16LE();
engine->freezeTime();
if (engine->_text->showDialogueBubble) {
engine->_redraw->drawBubble(otherActorIdx);
}
engine->_text->setFontCrossColor(engine->_scene->getActor(otherActorIdx)->talkColor);
engine->_scene->talkingActor = otherActorIdx;
engine->_text->drawTextFullscreen(textIdx);
engine->unfreezeTime();
engine->_redraw->redrawEngineActions(true);
return 0;
}
/**
* To increment the current chapter value
* @note Opcode @c 0x2D
*/
static int32 lINC_CHAPTER(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_gameState->gameChapter++;
return 0;
}
/**
* Found an object. (Parameter = Object Index)
* @note Opcode @c 0x2E
*/
static int32 lFOUND_OBJECT(TwinEEngine *engine, LifeScriptContext &ctx) {
const int32 item = ctx.stream.readByte();
engine->freezeTime();
engine->_gameState->processFoundItem(item);
engine->unfreezeTime();
engine->_redraw->redrawEngineActions(true);
return 0;
}
/**
* Set a new value to open the door (left way) (Parameter = distance to open).
* @note Opcode @c 0x2F
*/
static int32 lSET_DOOR_LEFT(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 distance = ctx.stream.readSint16LE();
ctx.actor->angle = ANGLE_270;
ctx.actor->x = ctx.actor->lastX - distance;
ctx.actor->dynamicFlags.bIsSpriteMoving = 0;
ctx.actor->speed = 0;
return 0;
}
/**
* Set a new value to open the door (right way) (Parameter = distance to open).
* @note Opcode @c 0x30
*/
static int32 lSET_DOOR_RIGHT(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 distance = ctx.stream.readSint16LE();
ctx.actor->angle = ANGLE_90;
ctx.actor->x = ctx.actor->lastX + distance;
ctx.actor->dynamicFlags.bIsSpriteMoving = 0;
ctx.actor->speed = 0;
return 0;
}
/**
* Set a new value to open the door (up way) (Parameter = distance to open).
* @note Opcode @c 0x31
*/
static int32 lSET_DOOR_UP(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 distance = ctx.stream.readSint16LE();
ctx.actor->angle = ANGLE_180;
ctx.actor->z = ctx.actor->lastZ - distance;
ctx.actor->dynamicFlags.bIsSpriteMoving = 0;
ctx.actor->speed = 0;
return 0;
}
/**
* Set a new value to open the door (down way) (Parameter = distance to open).
* @note Opcode @c 0x32
*/
static int32 lSET_DOOR_DOWN(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 distance = ctx.stream.readSint16LE();
ctx.actor->angle = ANGLE_0;
ctx.actor->z = ctx.actor->lastZ + distance;
ctx.actor->dynamicFlags.bIsSpriteMoving = 0;
ctx.actor->speed = 0;
return 0;
}
/**
* Give actor bonus. (Parameter = 0 (Don't change the actor bonus), > 0 (Change to another bonus))
* @note Opcode @c 0x33
*/
static int32 lGIVE_BONUS(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 flag = ctx.stream.readByte();
if (ctx.actor->bonusParameter.cloverleaf || ctx.actor->bonusParameter.kashes || ctx.actor->bonusParameter.key || ctx.actor->bonusParameter.lifepoints || ctx.actor->bonusParameter.magicpoints) {
engine->_actor->processActorExtraBonus(ctx.actorIdx);
}
if (flag != 0) {
ctx.actor->bonusParameter.unk1 = 1;
}
return 0;
}
/**
* Change to another room. (Parameter = Scene Index)
* @note Opcode @c 0x34
*/
static int32 lCHANGE_CUBE(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 sceneIdx = ctx.stream.readByte();
engine->_scene->needChangeScene = sceneIdx;
engine->_scene->heroPositionType = ScenePositionType::kScene;
return 0;
}
/**
* To set the current actor to collid with objects. (Parameter = 1(True) = other values(False))
* @note Opcode @c 0x35
*/
static int32 lOBJ_COL(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 collision = ctx.stream.readByte();
if (collision != 0) {
ctx.actor->staticFlags.bComputeCollisionWithObj = 1;
} else {
ctx.actor->staticFlags.bComputeCollisionWithObj = 0;
}
return 0;
}
/**
* To set the current actor to collid with bricks. (Parameter = 1(True), = 2(True and the actor is dead), = other values(False))
* @note Opcode @c 0x36
*/
static int32 lBRICK_COL(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 collision = ctx.stream.readByte();
ctx.actor->staticFlags.bComputeCollisionWithBricks = 0;
ctx.actor->staticFlags.bComputeLowCollision = 0;
if (collision == 1) {
ctx.actor->staticFlags.bComputeCollisionWithBricks = 1;
} else if (collision == 2) {
ctx.actor->staticFlags.bComputeCollisionWithBricks = 1;
ctx.actor->staticFlags.bComputeLowCollision = 1;
}
return 0;
}
/**
* To use various conditions for the same IF statement. (Use above an IF condition)
* @note Opcode @c 0x37
*/
static int32 lOR_IF(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 valueSize = processLifeConditions(engine, ctx);
if (processLifeOperators(engine, ctx, valueSize)) {
ctx.stream.seek(ctx.stream.readSint16LE()); // condition offset
} else {
ctx.stream.skip(2);
}
return 0;
}
/**
* Put an actor invisible (Parameter = 1(True), = 0(False))
* @note Opcode @c 0x38
*/
static int32 lINVISIBLE(TwinEEngine *engine, LifeScriptContext &ctx) {
ctx.actor->staticFlags.bIsHidden = ctx.stream.readByte();
return 0;
}
/**
* Camara zoom in and zoom out. (Parameter = 1(in) = 0(out))
* @note Opcode @c 0x39
*/
static int32 lZOOM(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->zoomScreen = ctx.stream.readByte();
if (engine->zoomScreen && !engine->_redraw->drawInGameTransBox && engine->cfgfile.SceZoom) {
engine->_screens->fadeToBlack(engine->_screens->mainPaletteRGBA);
engine->initMCGA();
engine->_screens->setBackPal();
engine->_screens->lockPalette = true;
} else if (!engine->zoomScreen && engine->_redraw->drawInGameTransBox) {
engine->_screens->fadeToBlack(engine->_screens->mainPaletteRGBA);
engine->initSVGA();
engine->_screens->setBackPal();
engine->_screens->lockPalette = true;
engine->_redraw->reqBgRedraw = true;
}
return 0;
}
/**
* Set new postion for the current actor (Parameter = Track Index)
* @note Opcode @c 0x3A
*/
static int32 lPOS_POINT(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 trackIdx = ctx.stream.readByte();
const ScenePoint &sp = engine->_scene->sceneTracks[trackIdx];
engine->_renderer->destX = sp.x;
engine->_renderer->destY = sp.y;
engine->_renderer->destZ = sp.z;
ctx.actor->x = sp.x;
ctx.actor->y = sp.y;
ctx.actor->z = sp.z;
return 0;
}
/**
* To set the magic level. (Paramater = Magic Level)
* @note Opcode @c 0x3B
*/
static int32 lSET_MAGIC_LEVEL(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_gameState->magicLevelIdx = ctx.stream.readByte();
engine->_gameState->inventoryMagicPoints = engine->_gameState->magicLevelIdx * 20;
return 0;
}
/**
* Substract the magic points. (Parameter = Points Value)
* @note Opcode @c 0x3C
*/
static int32 lSUB_MAGIC_POINT(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_gameState->inventoryMagicPoints = ctx.stream.readByte();
if (engine->_gameState->inventoryMagicPoints < 0) {
engine->_gameState->inventoryMagicPoints = 0;
}
return 0;
}
/**
* Set new a life point. (Parameter = Actor Index, Parameter = Points Value)
* @note Opcode @c 0x3D
*/
static int32 lSET_LIFE_POINT_OBJ(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 otherActorIdx = ctx.stream.readByte();
static int32 lifeValue = ctx.stream.readByte();
engine->_scene->getActor(otherActorIdx)->life = lifeValue;
return 0;
}
/**
* Substract the life points. (Parameter = Actor Index, Parameter = Points Value)
* @note Opcode @c 0x3E
*/
static int32 lSUB_LIFE_POINT_OBJ(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 otherActorIdx = ctx.stream.readByte();
static int32 lifeValue = ctx.stream.readByte();
ActorStruct *otherActor = engine->_scene->getActor(otherActorIdx);
otherActor->life -= lifeValue;
if (otherActor->life < 0) {
otherActor->life = 0;
}
return 0;
}
/**
* Hit an actor. (Parameter = Actor Index)
* @note Opcode @c 0x3F
*/
static int32 lHIT_OBJ(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 otherActorIdx = ctx.stream.readByte();
int32 strengthOfHit = ctx.stream.readByte();
engine->_actor->hitActor(ctx.actorIdx, otherActorIdx, strengthOfHit, engine->_scene->getActor(otherActorIdx)->angle);
return 0;
}
/**
* Play FLA cutscenes (Parameter = Cutscene Name)
* @note Opcode @c 0x40
*/
static int32 lPLAY_FLA(TwinEEngine *engine, LifeScriptContext &ctx) {
int strIdx = 0;
char movie[64];
do {
const byte c = ctx.stream.readByte();
movie[strIdx++] = c;
if (c == '\0') {
break;
}
if (strIdx >= ARRAYSIZE(movie)) {
error("Max string size exceeded for fla name");
}
} while (true);
engine->_flaMovies->playFlaMovie(movie);
engine->setPalette(engine->_screens->paletteRGBA);
engine->_screens->clearScreen();
engine->flip();
return 0;
}
/**
* Play Midis (Parameter = Midis Index)
* @note Opcode @c 0x41
*/
static int32 lPLAY_MIDI(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 midiIdx = ctx.stream.readByte();
engine->_music->playMidiMusic(midiIdx); // TODO: improve this
return 0;
}
/**
* To increment the clover box current value.
* @note Opcode @c 0x42
*/
static int32 lINC_CLOVER_BOX(TwinEEngine *engine, LifeScriptContext &ctx) {
if (engine->_gameState->inventoryNumLeafsBox < 10) {
engine->_gameState->inventoryNumLeafsBox++;
}
return 0;
}
/**
* To set an inventory object as used (Parameter = Object Index)
* @note Opcode @c 0x43
*/
static int32 lSET_USED_INVENTORY(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 item = ctx.stream.readByte();
if (item < InventoryItems::kKeypad) { // TODO: this looks wrong - why only up to keypad?
engine->_gameState->inventoryFlags[item] = 1;
}
return 0;
}
/**
* Add an option for the asked choice . (Parameter = Text Index in the current Text Bank)
* @note Opcode @c 0x44
*/
static int32 lADD_CHOICE(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 choiceIdx = ctx.stream.readSint16LE();
engine->_gameState->gameChoices[engine->_gameState->numChoices++] = choiceIdx;
return 0;
}
/**
* The current actor will ask something (parameter) with choices to choose. (Parameter = Text Index in the current Text Bank)
* @note Opcode @c 0x45
*/
static int32 lASK_CHOICE(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 choiceIdx = ctx.stream.readSint16LE();
engine->freezeTime();
if (engine->_text->showDialogueBubble) {
engine->_redraw->drawBubble(ctx.actorIdx);
}
engine->_text->setFontCrossColor(ctx.actor->talkColor);
engine->_gameState->processGameChoices(choiceIdx);
engine->_gameState->numChoices = 0;
engine->unfreezeTime();
engine->_redraw->redrawEngineActions(true);
return 0;
}
/**
* Show text in full screen. (Parameter = Text Index in the current Text Bank)
* @note Opcode @c 0x46
*/
static int32 lBIG_MESSAGE(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 textIdx = ctx.stream.readSint16LE();
engine->freezeTime();
engine->_text->textClipFull();
if (engine->_text->showDialogueBubble) {
engine->_redraw->drawBubble(ctx.actorIdx);
}
engine->_text->setFontCrossColor(ctx.actor->talkColor);
engine->_scene->talkingActor = ctx.actorIdx;
engine->_text->drawTextFullscreen(textIdx);
engine->_text->textClipSmall();
engine->unfreezeTime();
engine->_redraw->redrawEngineActions(true);
return 0;
}
/**
* To initiate the hidden meca-pingouin in the current scene. (Parameter = Actor Index)
* @note Opcode @c 0x47
*/
static int32 lINIT_PINGOUIN(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 pingouinActor = ctx.stream.readByte();
engine->_scene->mecaPinguinIdx = pingouinActor;
ActorStruct *mecaPinguin = engine->_scene->getActor(pingouinActor);
mecaPinguin->dynamicFlags.bIsDead = 1;
mecaPinguin->entity = -1;
mecaPinguin->zone = -1;
return 0;
}
/**
* To set an holomap position. (Parameter = Holomap/Scene Index)
* @note Opcode @c 0x48
*/
static int32 lSET_HOLO_POS(TwinEEngine *engine, LifeScriptContext &ctx) {
static int32 location = ctx.stream.readByte();
engine->_holomap->setHolomapPosition(location);
return 0;
}
/**
* To clear an holomap position. (Parameter = Holomap/Scene Index)
* @note Opcode @c 0x49
*/
static int32 lCLR_HOLO_POS(TwinEEngine *engine, LifeScriptContext &ctx) {
static int32 location = ctx.stream.readByte();
engine->_holomap->clearHolomapPosition(location);
return 0;
}
/**
* Add to the current fuel value the passed parameter. (Parameter = Fuel Amount)
* @note Opcode @c 0x4A
*/
static int32 lADD_FUEL(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_gameState->inventoryNumGas += ctx.stream.readByte();
if (engine->_gameState->inventoryNumGas > 100) {
engine->_gameState->inventoryNumGas = 100;
}
return 0;
}
/**
* Substract the to fuel value the value passed as parameter. (Parameter = Fuel Amount)
* @note Opcode @c 0x4B
*/
static int32 lSUB_FUEL(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_gameState->inventoryNumGas -= ctx.stream.readByte();
if (engine->_gameState->inventoryNumGas < 0) {
engine->_gameState->inventoryNumGas = 0;
}
return 0;
}
/**
* To set a GRID disappearing ceiling piece (Parameter = Disappearing ceiling piece Index)
* @note Opcode @c 0x4C
*/
static int32 lSET_GRM(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_grid->cellingGridIdx = ctx.stream.readByte();
engine->_grid->initCellingGrid(engine->_grid->cellingGridIdx);
return 0;
}
/**
* The current actor will say the message passed as paramenter. (Parameter = Actor Index)
* @note Opcode @c 0x4D
*/
static int32 lSAY_MESSAGE(TwinEEngine *engine, LifeScriptContext &ctx) {
int16 textEntry = ctx.stream.readSint16LE();
engine->_redraw->addOverlay(OverlayType::koText, textEntry, 0, 0, ctx.actorIdx, OverlayPosType::koFollowActor, 2);
ScopedEngineFreeze scoped(engine);
engine->_text->initVoxToPlay(textEntry);
return 0;
}
/**
* The actor passed as parameter will say the message passed as paramenter. (Parameter = Actor Index, Parameter = Text Index in the current Text Bank)
* @note Opcode @c 0x4E
*/
static int32 lSAY_MESSAGE_OBJ(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 otherActorIdx = ctx.stream.readByte();
int16 textEntry = ctx.stream.readSint16LE();
engine->_redraw->addOverlay(OverlayType::koText, textEntry, 0, 0, otherActorIdx, OverlayPosType::koFollowActor, 2);
ScopedEngineFreeze scoped(engine);
engine->_text->initVoxToPlay(textEntry);
return 0;
}
/**
* Set Twinsen life point as full.
* @note Opcode @c 0x4F
*/
static int32 lFULL_POINT(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_scene->sceneHero->life = 50;
engine->_gameState->inventoryMagicPoints = engine->_gameState->magicLevelIdx * 20;
return 0;
}
/**
* Change actor orientation. (Parameter = New Angle)
* @note Opcode @c 0x50
*/
static int32 lBETA(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 newAngle = ctx.stream.readSint16LE();
ctx.actor->angle = ToAngle(newAngle);
engine->_movements->clearRealAngle(ctx.actor);
return 0;
}
/**
* To unset the GRID disappearing ceiling piece.
* @note Opcode @c 0x51
*/
static int32 lGRM_OFF(TwinEEngine *engine, LifeScriptContext &ctx) {
if (engine->_grid->cellingGridIdx != -1) {
engine->_grid->useCellingGrid = -1;
engine->_grid->cellingGridIdx = -1;
engine->_grid->createGridMap();
engine->_redraw->redrawEngineActions(true);
}
return 0;
}
/**
* Fade palette to red
* @note Opcode @c 0x52
*/
static int32 lFADE_PAL_RED(TwinEEngine *engine, LifeScriptContext &ctx) {
ScopedEngineFreeze scoped(engine);
engine->_screens->fadePalRed(engine->_screens->mainPaletteRGBA);
engine->_screens->useAlternatePalette = false;
return 0;
}
/**
* Fade alarm to red
* @note Opcode @c 0x53
*/
static int32 lFADE_ALARM_RED(TwinEEngine *engine, LifeScriptContext &ctx) {
ScopedEngineFreeze scoped(engine);
HQR::getEntry(engine->_screens->palette, Resources::HQR_RESS_FILE, RESSHQR_ALARMREDPAL);
engine->_screens->convertPalToRGBA(engine->_screens->palette, engine->_screens->paletteRGBA);
engine->_screens->fadePalRed(engine->_screens->paletteRGBA);
engine->_screens->useAlternatePalette = true;
return 0;
}
/**
* Fade alarm to palette
* @note Opcode @c 0x54
*/
static int32 lFADE_ALARM_PAL(TwinEEngine *engine, LifeScriptContext &ctx) {
ScopedEngineFreeze scoped(engine);
HQR::getEntry(engine->_screens->palette, Resources::HQR_RESS_FILE, RESSHQR_ALARMREDPAL);
engine->_screens->convertPalToRGBA(engine->_screens->palette, engine->_screens->paletteRGBA);
engine->_screens->adjustCrossPalette(engine->_screens->paletteRGBA, engine->_screens->mainPaletteRGBA);
engine->_screens->useAlternatePalette = false;
return 0;
}
/**
* Fade red to palette
* @note Opcode @c 0x55
*/
static int32 lFADE_RED_PAL(TwinEEngine *engine, LifeScriptContext &ctx) {
ScopedEngineFreeze scoped(engine);
engine->_screens->fadeRedPal(engine->_screens->mainPaletteRGBA);
engine->_screens->useAlternatePalette = false;
return 0;
}
/**
* Fade red to alarm
* @note Opcode @c 0x56
*/
static int32 lFADE_RED_ALARM(TwinEEngine *engine, LifeScriptContext &ctx) {
ScopedEngineFreeze scoped(engine);
HQR::getEntry(engine->_screens->palette, Resources::HQR_RESS_FILE, RESSHQR_ALARMREDPAL);
engine->_screens->convertPalToRGBA(engine->_screens->palette, engine->_screens->paletteRGBA);
engine->_screens->fadeRedPal(engine->_screens->paletteRGBA);
engine->_screens->useAlternatePalette = true;
return 0;
}
/**
* Fade palette to alarm
* @note Opcode @c 0x57
*/
static int32 lFADE_PAL_ALARM(TwinEEngine *engine, LifeScriptContext &ctx) {
ScopedEngineFreeze scoped(engine);
HQR::getEntry(engine->_screens->palette, Resources::HQR_RESS_FILE, RESSHQR_ALARMREDPAL);
engine->_screens->convertPalToRGBA(engine->_screens->palette, engine->_screens->paletteRGBA);
engine->_screens->adjustCrossPalette(engine->_screens->mainPaletteRGBA, engine->_screens->paletteRGBA);
engine->_screens->useAlternatePalette = true;
return 0;
}
/**
* Explode an object. (Parameter = Object Index)
* @note Opcode @c 0x58
*/
static int32 lEXPLODE_OBJ(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 otherActorIdx = ctx.stream.readByte();
ActorStruct *otherActor = engine->_scene->getActor(otherActorIdx);
engine->_extra->addExtraExplode(otherActor->x, otherActor->y, otherActor->z); // RECHECK this
return 0;
}
/**
* Turn on bubbles while actors talk.
* @note Opcode @c 0x59
*/
static int32 lBUBBLE_ON(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_text->showDialogueBubble = true;
return 0;
}
/**
* Turn off bubbles while actors talk.
* @note Opcode @c 0x5A
*/
static int32 lBUBBLE_OFF(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_text->showDialogueBubble = true; // TODO: this looks wrong - why true and not false?
return 0;
}
/**
* The actor will ask something with choices to choose. (Parameter = Actor Index, Parameter = Text Index in the current Text Bank)
* @note Opcode @c 0x5B
*/
static int32 lASK_CHOICE_OBJ(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 otherActorIdx = ctx.stream.readByte();
int32 choiceIdx = ctx.stream.readSint16LE();
engine->freezeTime();
if (engine->_text->showDialogueBubble) {
engine->_redraw->drawBubble(otherActorIdx);
}
engine->_text->setFontCrossColor(engine->_scene->getActor(otherActorIdx)->talkColor);
engine->_gameState->processGameChoices(choiceIdx);
engine->_gameState->numChoices = 0;
engine->unfreezeTime();
engine->_redraw->redrawEngineActions(true);
return 0;
}
/**
* Set a dark palette.
* @note Opcode @c 0x5C
*/
static int32 lSET_DARK_PAL(TwinEEngine *engine, LifeScriptContext &ctx) {
ScopedEngineFreeze scoped(engine);
HQR::getEntry(engine->_screens->palette, Resources::HQR_RESS_FILE, RESSHQR_DARKPAL);
if (!engine->_screens->lockPalette) {
engine->_screens->convertPalToRGBA(engine->_screens->palette, engine->_screens->paletteRGBA);
engine->setPalette(engine->_screens->paletteRGBA);
}
engine->_screens->useAlternatePalette = true;
return 0;
}
/**
* Set main palette.
* @note Opcode @c 0x5D
*/
static int32 lSET_NORMAL_PAL(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_screens->useAlternatePalette = false;
if (!engine->_screens->lockPalette) {
engine->setPalette(engine->_screens->mainPaletteRGBA);
}
return 0;
}
/**
* Show Sendell message.
* @note Opcode @c 0x5E
*/
static int32 lMESSAGE_SENDELL(TwinEEngine *engine, LifeScriptContext &ctx) {
ScopedEngineFreeze scoped(engine);
engine->_screens->fadeToBlack(engine->_screens->paletteRGBA);
engine->_screens->loadImage(RESSHQR_TWINSEN_ZOE_SENDELLIMG, RESSHQR_TWINSEN_ZOE_SENDELLPAL);
engine->_text->textClipFull();
engine->_text->setFontCrossColor(15);
engine->_text->drawTextBoxBackground = false;
const bool tmpFlagDisplayText = engine->cfgfile.FlagDisplayText;
engine->cfgfile.FlagDisplayText = true;
engine->_text->drawTextFullscreen(TextId::kSendell);
engine->_text->drawTextBoxBackground = true;
engine->_text->textClipSmall();
engine->_screens->fadeToBlack(engine->_screens->paletteRGBACustom);
engine->_screens->clearScreen();
engine->setPalette(engine->_screens->paletteRGBA);
engine->cfgfile.FlagDisplayText = tmpFlagDisplayText;
return 0;
}
/**
* Set new animation for the current actor (Parameter = Animation Index)
* @note Opcode @c 0x5F
*/
static int32 lANIM_SET(TwinEEngine *engine, LifeScriptContext &ctx) {
AnimationTypes animIdx = (AnimationTypes)ctx.stream.readByte();
ctx.actor->anim = AnimationTypes::kAnimNone;
ctx.actor->previousAnimIdx = -1;
engine->_animations->initAnim(animIdx, 0, AnimationTypes::kStanding, ctx.actorIdx);
return 0;
}
/**
* Displays holomap travel animation. (Parameter = Trajectory)
* @note Opcode @c 0x60
*/
static int32 lHOLOMAP_TRAJ(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_holomap->drawHolomapTrajectory(ctx.stream.readByte());
return 0;
}
/**
* Game over.
* @note Opcode @c 0x61
*/
static int32 lGAME_OVER(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_scene->sceneHero->dynamicFlags.bAnimEnded = 1;
engine->_scene->sceneHero->life = 0;
engine->_gameState->inventoryNumLeafs = 0;
return 1; // break
}
/**
* End of the game.
* @note Opcode @c 0x62
*/
static int32 lTHE_END(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->quitGame = 1;
engine->_gameState->inventoryNumLeafs = 0;
engine->_scene->sceneHero->life = 50;
engine->_gameState->inventoryMagicPoints = 80;
engine->_scene->currentSceneIdx = LBA1SceneId::Polar_Island_Final_Battle;
engine->_actor->heroBehaviour = engine->_actor->previousHeroBehaviour;
engine->_scene->newHeroX = -1;
engine->_scene->sceneHero->angle = engine->_actor->previousHeroAngle;
engine->autoSave();
return 1; // break;
}
/**
* Stop the current played midi.
* @note Opcode @c 0x63
*/
static int32 lMIDI_OFF(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_music->stopMidiMusic();
return 0;
}
/**
* Play a CD Track (Paramenter = CD Track).
* @note Opcode @c 0x64
*/
static int32 lPLAY_CD_TRACK(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 track = ctx.stream.readByte();
engine->_music->playTrackMusic(track);
return 0;
}
/**
* Set isometric projections
* @note Opcode @c 0x65
*/
static int32 lPROJ_ISO(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_renderer->setOrthoProjection(311, 240, 512);
engine->_renderer->setBaseTranslation(0, 0, 0);
engine->_renderer->setBaseRotation(0, 0, 0);
engine->_renderer->setLightVector(engine->_scene->alphaLight, engine->_scene->betaLight, 0);
return 0;
}
/**
* Set 3D projections
* @note Opcode @c 0x66
*/
static int32 lPROJ_3D(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->_screens->copyScreen(engine->frontVideoBuffer, engine->workVideoBuffer);
engine->flip();
engine->_scene->changeRoomVar10 = false;
engine->_renderer->setCameraPosition(SCREEN_WIDTH / 2, SCREEN_HEIGHT / 2, 128, 1024, 1024);
engine->_renderer->setCameraAngle(0, 1500, 0, 25, -128, 0, 13000);
engine->_renderer->setLightVector(896, 950, ANGLE_0);
engine->_text->initTextBank(TextBankId::Credits);
return 0;
}
/**
* Only display the text. (e.g. like in the credit list) (Parameter = Text Index in the current Text Bank)
* @note Opcode @c 0x67
*/
static int32 lTEXT(TwinEEngine *engine, LifeScriptContext &ctx) {
int32 textIdx = ctx.stream.readSint16LE();
if (drawVar1 < 440) {
if (engine->cfgfile.Version == USA_VERSION) {
if (!textIdx) {
textIdx = TextId::kSaveSettings;
}
}
char textStr[256];
engine->_text->getMenuText(textIdx, textStr, sizeof(textStr));
int32 textSize = engine->_text->getTextSize(textStr);
int32 textBoxRight = textSize;
engine->_text->setFontColor(15);
engine->_text->drawText(0, drawVar1, textStr);
if (textSize > SCREEN_WIDTH - 1) {
textBoxRight = SCREEN_WIDTH - 1;
}
drawVar1 += 40;
// TODO: this looks wrong, top and bottom var are the same coordinates - the text height might be missing here
engine->copyBlockPhys(0, drawVar1, textBoxRight, drawVar1);
}
return 0;
}
/**
* Clear displayed text in the screen.
* @note Opcode @c 0x68
*/
static int32 lCLEAR_TEXT(TwinEEngine *engine, LifeScriptContext &ctx) {
drawVar1 = 0;
const Common::Rect rect(0, 0, SCREEN_WIDTH - 1, SCREEN_HEIGHT / 2);
engine->_interface->drawSplittedBox(rect, 0);
engine->copyBlockPhys(rect);
return 0;
}
/**
* Exit the script execution.
* @note Opcode @c 0x69
*/
static int32 lBRUTAL_EXIT(TwinEEngine *engine, LifeScriptContext &ctx) {
engine->quitGame = 0;
return 1; // break
}
static const ScriptLifeFunction function_map[] = {
/*0x00*/ MAPFUNC("END", lEND),
/*0x01*/ MAPFUNC("NOP", lNOP),
/*0x02*/ MAPFUNC("SNIF", lSNIF),
/*0x03*/ MAPFUNC("OFFSET", lOFFSET),
/*0x04*/ MAPFUNC("NEVERIF", lNEVERIF),
/*0x05*/ MAPFUNC("", lEMPTY), // unused
/*0x06*/ MAPFUNC("NO_IF", lNO_IF),
/*0x07*/ MAPFUNC("", lEMPTY), // unused
/*0x08*/ MAPFUNC("", lEMPTY), // unused
/*0x09*/ MAPFUNC("", lEMPTY), // unused
/*0x0A*/ MAPFUNC("LABEL", lLABEL),
/*0x0B*/ MAPFUNC("RETURN", lRETURN),
/*0x0C*/ MAPFUNC("IF", lIF),
/*0x0D*/ MAPFUNC("SWIF", lSWIF),
/*0x0E*/ MAPFUNC("ONEIF", lONEIF),
/*0x0F*/ MAPFUNC("ELSE", lELSE),
/*0x10*/ MAPFUNC("ENDIF", lEMPTY), // End of a conditional statement (e.g. IF)
/*0x11*/ MAPFUNC("BODY", lBODY),
/*0x12*/ MAPFUNC("BODY_OBJ", lBODY_OBJ),
/*0x13*/ MAPFUNC("ANIM", lANIM),
/*0x14*/ MAPFUNC("ANIM_OBJ", lANIM_OBJ),
/*0x15*/ MAPFUNC("SET_LIFE", lSET_LIFE),
/*0x16*/ MAPFUNC("SET_LIFE_OBJ", lSET_LIFE_OBJ),
/*0x17*/ MAPFUNC("SET_TRACK", lSET_TRACK),
/*0x18*/ MAPFUNC("SET_TRACK_OBJ", lSET_TRACK_OBJ),
/*0x19*/ MAPFUNC("MESSAGE", lMESSAGE),
/*0x1A*/ MAPFUNC("FALLABLE", lFALLABLE),
/*0x1B*/ MAPFUNC("SET_DIRMODE", lSET_DIRMODE),
/*0x1C*/ MAPFUNC("SET_DIRMODE_OBJ", lSET_DIRMODE_OBJ),
/*0x1D*/ MAPFUNC("CAM_FOLLOW", lCAM_FOLLOW),
/*0x1E*/ MAPFUNC("SET_BEHAVIOUR", lSET_BEHAVIOUR),
/*0x1F*/ MAPFUNC("SET_FLAG_CUBE", lSET_FLAG_CUBE),
/*0x20*/ MAPFUNC("COMPORTEMENT", lCOMPORTEMENT),
/*0x21*/ MAPFUNC("SET_COMPORTEMENT", lSET_COMPORTEMENT),
/*0x22*/ MAPFUNC("SET_COMPORTEMENT_OBJ", lSET_COMPORTEMENT_OBJ),
/*0x23*/ MAPFUNC("END_COMPORTEMENT", lEND_COMPORTEMENT),
/*0x24*/ MAPFUNC("SET_FLAG_GAME", lSET_FLAG_GAME),
/*0x25*/ MAPFUNC("KILL_OBJ", lKILL_OBJ),
/*0x26*/ MAPFUNC("SUICIDE", lSUICIDE),
/*0x27*/ MAPFUNC("USE_ONE_LITTLE_KEY", lUSE_ONE_LITTLE_KEY),
/*0x28*/ MAPFUNC("GIVE_GOLD_PIECES", lGIVE_GOLD_PIECES),
/*0x29*/ MAPFUNC("END_LIFE", lEND_LIFE),
/*0x2A*/ MAPFUNC("STOP_L_TRACK", lSTOP_L_TRACK),
/*0x2B*/ MAPFUNC("RESTORE_L_TRACK", lRESTORE_L_TRACK),
/*0x2C*/ MAPFUNC("MESSAGE_OBJ", lMESSAGE_OBJ),
/*0x2D*/ MAPFUNC("INC_CHAPTER", lINC_CHAPTER),
/*0x2E*/ MAPFUNC("FOUND_OBJECT", lFOUND_OBJECT),
/*0x2F*/ MAPFUNC("SET_DOOR_LEFT", lSET_DOOR_LEFT),
/*0x30*/ MAPFUNC("SET_DOOR_RIGHT", lSET_DOOR_RIGHT),
/*0x31*/ MAPFUNC("SET_DOOR_UP", lSET_DOOR_UP),
/*0x32*/ MAPFUNC("SET_DOOR_DOWN", lSET_DOOR_DOWN),
/*0x33*/ MAPFUNC("GIVE_BONUS", lGIVE_BONUS),
/*0x34*/ MAPFUNC("CHANGE_CUBE", lCHANGE_CUBE),
/*0x35*/ MAPFUNC("OBJ_COL", lOBJ_COL),
/*0x36*/ MAPFUNC("BRICK_COL", lBRICK_COL),
/*0x37*/ MAPFUNC("OR_IF", lOR_IF),
/*0x38*/ MAPFUNC("INVISIBLE", lINVISIBLE),
/*0x39*/ MAPFUNC("ZOOM", lZOOM),
/*0x3A*/ MAPFUNC("POS_POINT", lPOS_POINT),
/*0x3B*/ MAPFUNC("SET_MAGIC_LEVEL", lSET_MAGIC_LEVEL),
/*0x3C*/ MAPFUNC("SUB_MAGIC_POINT", lSUB_MAGIC_POINT),
/*0x3D*/ MAPFUNC("SET_LIFE_POINT_OBJ", lSET_LIFE_POINT_OBJ),
/*0x3E*/ MAPFUNC("SUB_LIFE_POINT_OBJ", lSUB_LIFE_POINT_OBJ),
/*0x3F*/ MAPFUNC("HIT_OBJ", lHIT_OBJ),
/*0x40*/ MAPFUNC("PLAY_FLA", lPLAY_FLA),
/*0x41*/ MAPFUNC("PLAY_MIDI", lPLAY_MIDI),
/*0x42*/ MAPFUNC("INC_CLOVER_BOX", lINC_CLOVER_BOX),
/*0x43*/ MAPFUNC("SET_USED_INVENTORY", lSET_USED_INVENTORY),
/*0x44*/ MAPFUNC("ADD_CHOICE", lADD_CHOICE),
/*0x45*/ MAPFUNC("ASK_CHOICE", lASK_CHOICE),
/*0x46*/ MAPFUNC("BIG_MESSAGE", lBIG_MESSAGE),
/*0x47*/ MAPFUNC("INIT_PINGOUIN", lINIT_PINGOUIN),
/*0x48*/ MAPFUNC("SET_HOLO_POS", lSET_HOLO_POS),
/*0x49*/ MAPFUNC("CLR_HOLO_POS", lCLR_HOLO_POS),
/*0x4A*/ MAPFUNC("ADD_FUEL", lADD_FUEL),
/*0x4B*/ MAPFUNC("SUB_FUEL", lSUB_FUEL),
/*0x4C*/ MAPFUNC("SET_GRM", lSET_GRM),
/*0x4D*/ MAPFUNC("SAY_MESSAGE", lSAY_MESSAGE),
/*0x4E*/ MAPFUNC("SAY_MESSAGE_OBJ", lSAY_MESSAGE_OBJ),
/*0x4F*/ MAPFUNC("FULL_POINT", lFULL_POINT),
/*0x50*/ MAPFUNC("BETA", lBETA),
/*0x51*/ MAPFUNC("GRM_OFF", lGRM_OFF),
/*0x52*/ MAPFUNC("FADE_PAL_RED", lFADE_PAL_RED),
/*0x53*/ MAPFUNC("FADE_ALARM_RED", lFADE_ALARM_RED),
/*0x54*/ MAPFUNC("FADE_ALARM_PAL", lFADE_ALARM_PAL),
/*0x55*/ MAPFUNC("FADE_RED_PAL", lFADE_RED_PAL),
/*0x56*/ MAPFUNC("FADE_RED_ALARM", lFADE_RED_ALARM),
/*0x57*/ MAPFUNC("FADE_PAL_ALARM", lFADE_PAL_ALARM),
/*0x58*/ MAPFUNC("EXPLODE_OBJ", lEXPLODE_OBJ),
/*0x59*/ MAPFUNC("BUBBLE_ON", lBUBBLE_ON),
/*0x5A*/ MAPFUNC("BUBBLE_OFF", lBUBBLE_OFF),
/*0x5B*/ MAPFUNC("ASK_CHOICE_OBJ", lASK_CHOICE_OBJ),
/*0x5C*/ MAPFUNC("SET_DARK_PAL", lSET_DARK_PAL),
/*0x5D*/ MAPFUNC("SET_NORMAL_PAL", lSET_NORMAL_PAL),
/*0x5E*/ MAPFUNC("MESSAGE_SENDELL", lMESSAGE_SENDELL),
/*0x5F*/ MAPFUNC("ANIM_SET", lANIM_SET),
/*0x60*/ MAPFUNC("HOLOMAP_TRAJ", lHOLOMAP_TRAJ),
/*0x61*/ MAPFUNC("GAME_OVER", lGAME_OVER),
/*0x62*/ MAPFUNC("THE_END", lTHE_END),
/*0x63*/ MAPFUNC("MIDI_OFF", lMIDI_OFF),
/*0x64*/ MAPFUNC("PLAY_CD_TRACK", lPLAY_CD_TRACK),
/*0x65*/ MAPFUNC("PROJ_ISO", lPROJ_ISO),
/*0x66*/ MAPFUNC("PROJ_3D", lPROJ_3D),
/*0x67*/ MAPFUNC("TEXT", lTEXT),
/*0x68*/ MAPFUNC("CLEAR_TEXT", lCLEAR_TEXT),
/*0x69*/ MAPFUNC("BRUTAL_EXIT", lBRUTAL_EXIT)};
ScriptLife::ScriptLife(TwinEEngine *engine) : _engine(engine) {
drawVar1 = 0;
}
void ScriptLife::processLifeScript(int32 actorIdx) {
ActorStruct *actor = _engine->_scene->getActor(actorIdx);
int32 end = -2;
LifeScriptContext ctx(actorIdx, actor);
do {
const byte scriptOpcode = ctx.stream.readByte();
if (scriptOpcode < ARRAYSIZE(function_map)) {
end = function_map[scriptOpcode].function(_engine, ctx);
} else {
error("Actor %d with wrong offset/opcode - Offset: %d (opcode: %i)", actorIdx, ctx.stream.pos() - 1, scriptOpcode);
}
if (end < 0) {
warning("Actor %d Life script [%s] not implemented", actorIdx, function_map[scriptOpcode].name);
}
ctx.updateOpcodePos();
} while (end != 1);
}
} // namespace TwinE