2020-10-14 15:16:30 +02: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.
|
|
|
|
*
|
|
|
|
* 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/actor.h"
|
|
|
|
#include "twine/animations.h"
|
|
|
|
#include "twine/extra.h"
|
|
|
|
#include "twine/gamestate.h"
|
|
|
|
#include "twine/grid.h"
|
|
|
|
#include "twine/movements.h"
|
|
|
|
#include "twine/music.h"
|
|
|
|
#include "twine/redraw.h"
|
|
|
|
#include "twine/renderer.h"
|
|
|
|
#include "twine/resources.h"
|
|
|
|
#include "twine/scene.h"
|
|
|
|
#include "twine/screens.h"
|
|
|
|
#include "twine/sound.h"
|
|
|
|
#include "twine/text.h"
|
|
|
|
#include "twine/twine.h"
|
|
|
|
|
|
|
|
namespace TwinE {
|
|
|
|
|
|
|
|
void Scene::setActorStaticFlags(int32 actorIdx, uint16 staticFlags) {
|
2020-10-14 14:20:38 +02:00
|
|
|
if (staticFlags & 0x1) {
|
|
|
|
sceneActors[actorIdx].staticFlags.bComputeCollisionWithObj = 1;
|
|
|
|
}
|
|
|
|
if (staticFlags & 0x2) {
|
|
|
|
sceneActors[actorIdx].staticFlags.bComputeCollisionWithBricks = 1;
|
|
|
|
}
|
|
|
|
if (staticFlags & 0x4) {
|
|
|
|
sceneActors[actorIdx].staticFlags.bIsZonable = 1;
|
|
|
|
}
|
|
|
|
if (staticFlags & 0x8) {
|
|
|
|
sceneActors[actorIdx].staticFlags.bUsesClipping = 1;
|
|
|
|
}
|
|
|
|
if (staticFlags & 0x10) {
|
|
|
|
sceneActors[actorIdx].staticFlags.bCanBePushed = 1;
|
|
|
|
}
|
|
|
|
if (staticFlags & 0x20) {
|
|
|
|
sceneActors[actorIdx].staticFlags.bComputeLowCollision = 1;
|
|
|
|
}
|
|
|
|
if (staticFlags & 0x40) {
|
|
|
|
sceneActors[actorIdx].staticFlags.bCanDrown = 1;
|
|
|
|
}
|
|
|
|
if (staticFlags & 0x80) {
|
|
|
|
sceneActors[actorIdx].staticFlags.bUnk80 = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (staticFlags & 0x100) {
|
|
|
|
sceneActors[actorIdx].staticFlags.bUnk0100 = 1;
|
|
|
|
}
|
|
|
|
if (staticFlags & 0x200) {
|
|
|
|
sceneActors[actorIdx].staticFlags.bIsHidden = 1;
|
|
|
|
}
|
|
|
|
if (staticFlags & 0x400) {
|
|
|
|
sceneActors[actorIdx].staticFlags.bIsSpriteActor = 1;
|
|
|
|
}
|
|
|
|
if (staticFlags & 0x800) {
|
|
|
|
sceneActors[actorIdx].staticFlags.bCanFall = 1;
|
|
|
|
}
|
|
|
|
if (staticFlags & 0x1000) {
|
|
|
|
sceneActors[actorIdx].staticFlags.bDoesntCastShadow = 1;
|
|
|
|
}
|
|
|
|
if (staticFlags & 0x2000) {
|
|
|
|
//sceneActors[actorIdx].staticFlags.bIsBackgrounded = 1;
|
|
|
|
}
|
|
|
|
if (staticFlags & 0x4000) {
|
|
|
|
sceneActors[actorIdx].staticFlags.bIsCarrierActor = 1;
|
|
|
|
}
|
|
|
|
if (staticFlags & 0x8000) {
|
|
|
|
sceneActors[actorIdx].staticFlags.bUseMiniZv = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
void Scene::loadScene() {
|
2020-10-14 14:20:38 +02:00
|
|
|
int32 i;
|
|
|
|
int32 scriptSize = 0;
|
2020-10-14 15:16:30 +02:00
|
|
|
uint8 *localScene = currentScene;
|
2020-10-14 14:20:38 +02:00
|
|
|
|
|
|
|
// load scene ambience properties
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_text->currentTextBank = *(localScene++);
|
2020-10-14 14:20:38 +02:00
|
|
|
currentGameOverScene = *(localScene++);
|
|
|
|
localScene += 4;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
alphaLight = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
betaLight = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
|
|
|
|
// FIXME: Workaround to fix lighting issue - not using proper dark light
|
|
|
|
alphaLight = 896;
|
|
|
|
betaLight = 950;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
sampleAmbiance[0] = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sampleRepeat[0] = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sampleRound[0] = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
sampleAmbiance[1] = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sampleRepeat[1] = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sampleRound[1] = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
sampleAmbiance[2] = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sampleRepeat[2] = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sampleRound[2] = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
sampleAmbiance[3] = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sampleRepeat[3] = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sampleRound[3] = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
sampleMinDelay = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sampleMinDelayRnd = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
|
|
|
|
sceneMusic = *(localScene++);
|
|
|
|
|
|
|
|
// load hero properties
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneHeroX = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneHeroY = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneHeroZ = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
scriptSize = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
sceneHero->moveScript = localScene;
|
|
|
|
localScene += scriptSize;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
scriptSize = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
sceneHero->lifeScript = localScene;
|
|
|
|
localScene += scriptSize;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneNumActors = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
|
|
|
|
for (i = 1; i < sceneNumActors; i++) {
|
|
|
|
uint16 staticFlags;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_actor->resetActor(i);
|
2020-10-14 14:20:38 +02:00
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
staticFlags = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
setActorStaticFlags(i, staticFlags);
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneActors[i].entity = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
|
|
|
|
if (!sceneActors[i].staticFlags.bIsSpriteActor) {
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_hqrdepack->hqrGetallocEntry(&sceneActors[i].entityDataPtr, Resources::HQR_FILE3D_FILE, sceneActors[i].entity);
|
2020-10-14 14:20:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
sceneActors[i].body = *(localScene++);
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneActors[i].anim = (AnimationTypes)*(localScene++);
|
|
|
|
sceneActors[i].sprite = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneActors[i].x = *((uint16 *)localScene);
|
|
|
|
sceneActors[i].collisionX = sceneActors[i].x;
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneActors[i].y = *((uint16 *)localScene);
|
|
|
|
sceneActors[i].collisionY = sceneActors[i].y;
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneActors[i].z = *((uint16 *)localScene);
|
|
|
|
sceneActors[i].collisionZ = sceneActors[i].z;
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
sceneActors[i].strengthOfHit = *(localScene++);
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneActors[i].bonusParameter = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
sceneActors[i].bonusParameter &= 0xFE;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneActors[i].angle = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneActors[i].speed = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneActors[i].controlMode = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneActors[i].info0 = *((int16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneActors[i].info1 = *((int16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneActors[i].info2 = *((int16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneActors[i].info3 = *((int16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
sceneActors[i].followedActor = sceneActors[i].info3;
|
|
|
|
sceneActors[i].bonusAmount = *(localScene++);
|
|
|
|
sceneActors[i].talkColor = *(localScene++);
|
|
|
|
sceneActors[i].armor = *(localScene++);
|
|
|
|
sceneActors[i].life = *(localScene++);
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
scriptSize = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
sceneActors[i].moveScript = localScene;
|
|
|
|
localScene += scriptSize;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
scriptSize = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
sceneActors[i].lifeScript = localScene;
|
|
|
|
localScene += scriptSize;
|
|
|
|
}
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneNumZones = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
|
|
|
|
for (i = 0; i < sceneNumZones; i++) {
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneZones[i].bottomLeft.x = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneZones[i].bottomLeft.y = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneZones[i].bottomLeft.z = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneZones[i].topRight.x = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneZones[i].topRight.y = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneZones[i].topRight.z = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneZones[i].type = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneZones[i].infoData.generic.info0 = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneZones[i].infoData.generic.info1 = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneZones[i].infoData.generic.info2 = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneZones[i].infoData.generic.info3 = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneZones[i].snap = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
}
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneNumTracks = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
|
|
|
|
for (i = 0; i < sceneNumTracks; i++) {
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneTracks[i].x = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneTracks[i].y = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneTracks[i].z = *((uint16 *)localScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
localScene += 2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
int32 Scene::initScene(int32 index) {
|
2020-10-14 14:20:38 +02:00
|
|
|
// load scene from file
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_hqrdepack->hqrGetallocEntry(¤tScene, Resources::HQR_SCENE_FILE, index);
|
2020-10-14 14:20:38 +02:00
|
|
|
|
|
|
|
loadScene();
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
void Scene::resetScene() {
|
2020-10-14 14:20:38 +02:00
|
|
|
int32 i;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_extra->resetExtras();
|
2020-10-14 14:20:38 +02:00
|
|
|
|
|
|
|
for (i = 0; i < NUM_SCENES_FLAGS; i++) {
|
|
|
|
sceneFlags[i] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < OVERLAY_MAX_ENTRIES; i++) {
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_redraw->overlayList[i].info0 = -1;
|
2020-10-14 14:20:38 +02:00
|
|
|
}
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_actor->currentPositionInBodyPtrTab = 0;
|
2020-10-22 12:42:57 +02:00
|
|
|
_engine->_screens->useAlternatePalette = false;
|
2020-10-14 14:20:38 +02:00
|
|
|
}
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
void Scene::changeScene() {
|
2020-10-14 14:20:38 +02:00
|
|
|
int32 a;
|
|
|
|
|
|
|
|
// change twinsen house destroyed hard-coded
|
2020-10-14 15:16:30 +02:00
|
|
|
if (needChangeScene == 4 && _engine->_gameState->gameFlags[30] != 0)
|
2020-10-14 14:20:38 +02:00
|
|
|
needChangeScene = 118;
|
|
|
|
|
|
|
|
// local backup previous scene
|
|
|
|
previousSceneIdx = currentSceneIdx;
|
|
|
|
currentSceneIdx = needChangeScene;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_sound->stopSamples();
|
2020-10-14 14:20:38 +02:00
|
|
|
|
|
|
|
resetScene();
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_actor->loadHeroEntities();
|
2020-10-14 14:20:38 +02:00
|
|
|
|
|
|
|
sceneHero->controlMode = 1;
|
|
|
|
sceneHero->zone = -1;
|
|
|
|
sceneHero->positionInLifeScript = 0;
|
|
|
|
sceneHero->positionInMoveScript = -1;
|
|
|
|
sceneHero->labelIdx = -1;
|
|
|
|
|
|
|
|
initScene(needChangeScene);
|
|
|
|
|
|
|
|
//TODO: treat holomap trajectories
|
|
|
|
|
|
|
|
if (needChangeScene == 116 || needChangeScene == 117)
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_text->currentTextBank = 10;
|
2020-10-14 14:20:38 +02:00
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_text->initTextBank(_engine->_text->currentTextBank + 3);
|
|
|
|
_engine->_grid->initGrid(needChangeScene);
|
2020-10-14 14:20:38 +02:00
|
|
|
|
|
|
|
if (heroPositionType == kZone) {
|
|
|
|
newHeroX = zoneHeroX;
|
|
|
|
newHeroY = zoneHeroY;
|
|
|
|
newHeroZ = zoneHeroZ;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (heroPositionType == kScene || heroPositionType == kNoPosition) {
|
|
|
|
newHeroX = sceneHeroX;
|
|
|
|
newHeroY = sceneHeroY;
|
|
|
|
newHeroZ = sceneHeroZ;
|
|
|
|
}
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
sceneHero->x = newHeroX;
|
|
|
|
sceneHero->y = heroYBeforeFall = newHeroY;
|
|
|
|
sceneHero->z = newHeroZ;
|
2020-10-14 14:20:38 +02:00
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_renderer->setLightVector(alphaLight, betaLight, 0);
|
2020-10-14 14:20:38 +02:00
|
|
|
|
|
|
|
if (previousSceneIdx != needChangeScene) {
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_actor->previousHeroBehaviour = _engine->_actor->heroBehaviour;
|
|
|
|
_engine->_actor->previousHeroAngle = sceneHero->angle;
|
|
|
|
_engine->_gameState->saveGame();
|
2020-10-14 14:20:38 +02:00
|
|
|
}
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_actor->restartHeroScene();
|
2020-10-14 14:20:38 +02:00
|
|
|
|
|
|
|
for (a = 1; a < sceneNumActors; a++) {
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_actor->initActor(a);
|
2020-10-14 14:20:38 +02:00
|
|
|
}
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_gameState->inventoryNumKeys = 0;
|
|
|
|
_engine->disableScreenRecenter = 0;
|
2020-10-14 14:20:38 +02:00
|
|
|
heroPositionType = kNoPosition;
|
|
|
|
sampleAmbienceTime = 0;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_grid->newCameraX = sceneActors[currentlyFollowedActor].x >> 9;
|
|
|
|
_engine->_grid->newCameraY = sceneActors[currentlyFollowedActor].y >> 8;
|
|
|
|
_engine->_grid->newCameraZ = sceneActors[currentlyFollowedActor].z >> 9;
|
2020-10-14 14:20:38 +02:00
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_gameState->magicBallIdx = -1;
|
|
|
|
_engine->_movements->heroMoved = 1;
|
|
|
|
_engine->_grid->useCellingGrid = -1;
|
|
|
|
_engine->_grid->cellingGridIdx = -1;
|
2020-10-22 12:42:57 +02:00
|
|
|
_engine->_redraw->reqBgRedraw = true;
|
|
|
|
_engine->_screens->lockPalette = false;
|
2020-10-14 14:20:38 +02:00
|
|
|
|
|
|
|
needChangeScene = -1;
|
|
|
|
changeRoomVar10 = 1;
|
|
|
|
changeRoomVar11 = 14;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_renderer->setLightVector(alphaLight, betaLight, 0);
|
2020-10-14 14:20:38 +02:00
|
|
|
|
|
|
|
if (sceneMusic != -1) {
|
2020-10-22 12:27:06 +02:00
|
|
|
_engine->_music->playMidiMusic(sceneMusic); // TODO this should play midi or cd tracks
|
2020-10-14 14:20:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
void Scene::processEnvironmentSound() {
|
2020-10-14 14:20:38 +02:00
|
|
|
int16 s, currentAmb, decal, repeat;
|
|
|
|
int16 sampleIdx = -1;
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
if (_engine->lbaTime >= sampleAmbienceTime) {
|
|
|
|
currentAmb = _engine->getRandomNumber(4); // random ambiance
|
2020-10-14 14:20:38 +02:00
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
for (s = 0; s < 4; s++) {
|
|
|
|
if (!(samplePlayed & (1 << currentAmb))) { // if not already played
|
|
|
|
samplePlayed |= (1 << currentAmb); // make sample played
|
2020-10-14 14:20:38 +02:00
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
if (samplePlayed == 15) { // reset if all samples played
|
2020-10-14 14:20:38 +02:00
|
|
|
samplePlayed = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
sampleIdx = sampleAmbiance[currentAmb];
|
2020-10-14 15:16:30 +02:00
|
|
|
if (sampleIdx != -1) {
|
2020-10-14 14:20:38 +02:00
|
|
|
decal = sampleRound[currentAmb];
|
|
|
|
repeat = sampleRepeat[currentAmb];
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_sound->playSample(sampleIdx, (0x1000 + _engine->getRandomNumber(decal) - (decal / 2)), repeat, 110, -1, 110, -1);
|
|
|
|
break;
|
2020-10-14 14:20:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
currentAmb++; // try next ambiance
|
|
|
|
currentAmb &= 3; // loop in all 4 ambiances
|
|
|
|
}
|
|
|
|
|
|
|
|
// compute next ambiance timer
|
2020-10-14 15:16:30 +02:00
|
|
|
sampleAmbienceTime = _engine->lbaTime + (_engine->getRandomNumber(sampleMinDelayRnd) + sampleMinDelay) * 50;
|
2020-10-14 14:20:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
void Scene::processZoneExtraBonus(ZoneStruct *zone) {
|
2020-10-14 14:20:38 +02:00
|
|
|
int32 a, numBonus;
|
|
|
|
int8 bonusTable[8], currentBonus;
|
|
|
|
|
|
|
|
numBonus = 0;
|
|
|
|
|
|
|
|
// bonus not used yet
|
|
|
|
if (!zone->infoData.generic.info3) {
|
|
|
|
for (a = 0; a < 5; a++) {
|
|
|
|
if (zone->infoData.generic.info1 & (1 << (a + 4))) {
|
|
|
|
bonusTable[numBonus++] = a;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (numBonus) {
|
|
|
|
int32 angle, index;
|
2020-10-14 15:16:30 +02:00
|
|
|
currentBonus = bonusTable[_engine->getRandomNumber(numBonus)];
|
2020-10-14 14:20:38 +02:00
|
|
|
|
|
|
|
// if bonus is magic an no magic level yet, then give life points
|
2020-10-14 15:16:30 +02:00
|
|
|
if (!_engine->_gameState->magicLevelIdx && currentBonus == 2) {
|
2020-10-14 14:20:38 +02:00
|
|
|
currentBonus = 1;
|
|
|
|
}
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
angle = _engine->_movements->getAngleAndSetTargetActorDistance(ABS(zone->topRight.x + zone->bottomLeft.x) / 2, ABS(zone->topRight.z + zone->bottomLeft.z) / 2, sceneHero->x, sceneHero->z);
|
|
|
|
index = _engine->_extra->addExtraBonus(ABS(zone->topRight.x + zone->bottomLeft.x) / 2, zone->topRight.y, ABS(zone->topRight.z + zone->bottomLeft.z) / 2, 180, angle, currentBonus + 3, zone->infoData.generic.info2);
|
2020-10-14 14:20:38 +02:00
|
|
|
|
|
|
|
if (index != -1) {
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_extra->extraList[index].type |= 0x400;
|
2020-10-14 14:20:38 +02:00
|
|
|
zone->infoData.generic.info3 = 1; // set as used
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
void Scene::processActorZones(int32 actorIdx) {
|
2020-10-14 14:20:38 +02:00
|
|
|
int32 currentX, currentY, currentZ, z, tmpCellingGrid;
|
|
|
|
ActorStruct *actor;
|
|
|
|
|
|
|
|
actor = &sceneActors[actorIdx];
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
currentX = actor->x;
|
|
|
|
currentY = actor->y;
|
|
|
|
currentZ = actor->z;
|
2020-10-14 14:20:38 +02:00
|
|
|
|
|
|
|
actor->zone = -1;
|
|
|
|
tmpCellingGrid = 0;
|
|
|
|
|
|
|
|
if (!actorIdx) {
|
|
|
|
currentActorInZone = actorIdx;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (z = 0; z < sceneNumZones; z++) {
|
|
|
|
ZoneStruct *zone = &sceneZones[z];
|
|
|
|
|
|
|
|
// check if actor is in zone
|
2020-10-14 15:16:30 +02:00
|
|
|
if ((currentX >= zone->bottomLeft.x && currentX <= zone->topRight.x) &&
|
|
|
|
(currentY >= zone->bottomLeft.y && currentY <= zone->topRight.y) &&
|
|
|
|
(currentZ >= zone->bottomLeft.z && currentZ <= zone->topRight.z)) {
|
2020-10-14 14:20:38 +02:00
|
|
|
switch (zone->type) {
|
|
|
|
case kCube:
|
|
|
|
if (!actorIdx && actor->life > 0) {
|
|
|
|
needChangeScene = zone->infoData.ChangeScene.newSceneIdx;
|
2020-10-14 15:16:30 +02:00
|
|
|
zoneHeroX = actor->x - zone->bottomLeft.x + zone->infoData.ChangeScene.x;
|
|
|
|
zoneHeroY = actor->y - zone->bottomLeft.y + zone->infoData.ChangeScene.y;
|
|
|
|
zoneHeroZ = actor->z - zone->bottomLeft.z + zone->infoData.ChangeScene.z;
|
2020-10-14 14:20:38 +02:00
|
|
|
heroPositionType = kZone;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case kCamera:
|
|
|
|
if (currentlyFollowedActor == actorIdx) {
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->disableScreenRecenter = 1;
|
|
|
|
if (_engine->_grid->newCameraX != zone->infoData.CameraView.x || _engine->_grid->newCameraY != zone->infoData.CameraView.y || _engine->_grid->newCameraZ != zone->infoData.CameraView.z) {
|
|
|
|
_engine->_grid->newCameraX = zone->infoData.CameraView.x;
|
|
|
|
_engine->_grid->newCameraY = zone->infoData.CameraView.y;
|
|
|
|
_engine->_grid->newCameraZ = zone->infoData.CameraView.z;
|
2020-10-22 12:42:57 +02:00
|
|
|
_engine->_redraw->reqBgRedraw = true;
|
2020-10-14 14:20:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case kSceneric:
|
|
|
|
actor->zone = zone->infoData.Sceneric.zoneIdx;
|
|
|
|
break;
|
|
|
|
case kGrid:
|
|
|
|
if (currentlyFollowedActor == actorIdx) {
|
|
|
|
tmpCellingGrid = 1;
|
2020-10-14 15:16:30 +02:00
|
|
|
if (_engine->_grid->useCellingGrid != zone->infoData.CeillingGrid.newGrid) {
|
2020-10-14 14:20:38 +02:00
|
|
|
if (zone->infoData.CeillingGrid.newGrid != -1) {
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_grid->createGridMap();
|
2020-10-14 14:20:38 +02:00
|
|
|
}
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_grid->useCellingGrid = zone->infoData.CeillingGrid.newGrid;
|
|
|
|
_engine->_grid->cellingGridIdx = z;
|
|
|
|
_engine->freezeTime();
|
|
|
|
_engine->_grid->initCellingGrid(_engine->_grid->useCellingGrid);
|
|
|
|
_engine->unfreezeTime();
|
2020-10-14 14:20:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case kObject:
|
2020-10-14 15:16:30 +02:00
|
|
|
if (!actorIdx && _engine->_movements->heroAction != 0) {
|
|
|
|
_engine->_animations->initAnim(kAction, 1, 0, 0);
|
2020-10-14 14:20:38 +02:00
|
|
|
processZoneExtraBonus(zone);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case kText:
|
2020-10-14 15:16:30 +02:00
|
|
|
if (!actorIdx && _engine->_movements->heroAction != 0) {
|
|
|
|
_engine->freezeTime();
|
|
|
|
_engine->_text->setFontCrossColor(zone->infoData.DisplayText.textColor);
|
2020-10-14 14:20:38 +02:00
|
|
|
talkingActor = actorIdx;
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_text->drawTextFullscreen(zone->infoData.DisplayText.textIdx);
|
|
|
|
_engine->unfreezeTime();
|
|
|
|
_engine->_redraw->redrawEngineActions(1);
|
2020-10-14 14:20:38 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case kLadder:
|
2020-10-14 15:16:30 +02:00
|
|
|
if (!actorIdx && _engine->_actor->heroBehaviour != kProtoPack && (actor->anim == kForward || actor->anim == kTopLadder || actor->anim == kClimbLadder)) {
|
|
|
|
_engine->_movements->rotateActor(actor->boudingBox.x.bottomLeft, actor->boudingBox.z.bottomLeft, actor->angle + 0x580);
|
|
|
|
_engine->_renderer->destX += _engine->_movements->processActorX;
|
|
|
|
_engine->_renderer->destZ += _engine->_movements->processActorZ;
|
2020-10-14 14:20:38 +02:00
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
if (_engine->_renderer->destX >= 0 && _engine->_renderer->destZ >= 0 && _engine->_renderer->destX <= 0x7E00 && _engine->_renderer->destZ <= 0x7E00) {
|
|
|
|
if (_engine->_grid->getBrickShape(_engine->_renderer->destX, actor->y + 0x100, _engine->_renderer->destZ)) {
|
2020-10-14 14:20:38 +02:00
|
|
|
currentActorInZone = 1;
|
2020-10-14 15:16:30 +02:00
|
|
|
if (actor->y >= ABS(zone->bottomLeft.y + zone->topRight.y) / 2) {
|
|
|
|
_engine->_animations->initAnim(kTopLadder, 2, 0, actorIdx); // reached end of ladder
|
2020-10-14 14:20:38 +02:00
|
|
|
} else {
|
2020-10-14 15:16:30 +02:00
|
|
|
_engine->_animations->initAnim(kClimbLadder, 0, 255, actorIdx); // go up in ladder
|
2020-10-14 14:20:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-14 15:16:30 +02:00
|
|
|
if (!tmpCellingGrid && actorIdx == currentlyFollowedActor && _engine->_grid->useCellingGrid != -1) {
|
|
|
|
_engine->_grid->useCellingGrid = -1;
|
|
|
|
_engine->_grid->cellingGridIdx = -1;
|
|
|
|
_engine->_grid->createGridMap();
|
2020-10-22 12:42:57 +02:00
|
|
|
_engine->_redraw->reqBgRedraw = true;
|
2020-10-14 14:20:38 +02:00
|
|
|
}
|
|
|
|
}
|
2020-10-14 15:16:30 +02:00
|
|
|
|
|
|
|
} // namespace TwinE
|