mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-06 09:48:39 +00:00
2868 lines
76 KiB
C++
2868 lines
76 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 "asylum/views/scene.h"
|
|
|
|
#include "asylum/resources/actor.h"
|
|
#include "asylum/resources/encounters.h"
|
|
#include "asylum/resources/inventory.h"
|
|
#include "asylum/resources/object.h"
|
|
#include "asylum/resources/polygons.h"
|
|
#include "asylum/resources/script.h"
|
|
#include "asylum/resources/special.h"
|
|
#include "asylum/resources/worldstats.h"
|
|
|
|
#include "asylum/system/cursor.h"
|
|
#include "asylum/system/graphics.h"
|
|
#include "asylum/system/savegame.h"
|
|
#include "asylum/system/screen.h"
|
|
#include "asylum/system/speech.h"
|
|
#include "asylum/system/text.h"
|
|
|
|
#include "asylum/views/menu.h"
|
|
#include "asylum/views/scenetitle.h"
|
|
|
|
#include "asylum/asylum.h"
|
|
#include "asylum/respack.h"
|
|
#include "asylum/staticres.h"
|
|
|
|
namespace Asylum {
|
|
|
|
#define SCREEN_EDGES 40
|
|
#define SCROLL_STEP 10
|
|
|
|
int g_debugActors;
|
|
int g_debugObjects;
|
|
int g_debugPolygons;
|
|
int g_debugSceneRects;
|
|
int g_debugScrolling;
|
|
|
|
Scene::Scene(AsylumEngine *engine): _vm(engine),
|
|
_polygons(NULL), _ws(NULL) {
|
|
|
|
// Initialize data
|
|
_packId = kResourcePackInvalid;
|
|
|
|
_hitAreaChapter7Counter = 0;
|
|
_isCTRLPressed = false;
|
|
_chapter5RainFrameIndex = 0;
|
|
|
|
_musicVolume = 0;
|
|
_frameCounter = 0;
|
|
|
|
g_debugActors = 0;
|
|
g_debugObjects = 0;
|
|
g_debugPolygons = 0;
|
|
g_debugSceneRects = 0;
|
|
g_debugScrolling = 0;
|
|
}
|
|
|
|
Scene::~Scene() {
|
|
// Unload the associated resources
|
|
getResource()->unload(_packId);
|
|
|
|
// Clear script queue
|
|
getScript()->reset();
|
|
|
|
delete _polygons;
|
|
delete _ws;
|
|
}
|
|
|
|
void Scene::enter(ResourcePackId packId) {
|
|
_vm->setGameFlag(kGameFlagScriptProcessing);
|
|
|
|
getCursor()->hide();
|
|
|
|
getSharedData()->setPlayerIndex(0);
|
|
|
|
// Load the scene data
|
|
load(packId);
|
|
|
|
// Set wheel indices
|
|
_ws->setWheelObjects();
|
|
|
|
// Adjust object priority
|
|
if (_ws->objects.size() > 0) {
|
|
int32 priority = 4091;
|
|
|
|
for (uint32 i = 0; i < _ws->objects.size(); i++) {
|
|
Object *object = _ws->objects[i];
|
|
object->setPriority(priority);
|
|
object->flags &= ~kObjectFlagC000;
|
|
priority -= 4;
|
|
}
|
|
}
|
|
|
|
// Set the cursor to magnifying glass
|
|
getCursor()->set(_ws->cursorResources[kCursorResourceMagnifyingGlass], 0, kCursorAnimationNone);
|
|
getCursor()->show();
|
|
|
|
// Clear the graphic queue
|
|
getScreen()->clearGraphicsInQueue();
|
|
|
|
_ws->sceneRectIdx = 0;
|
|
_ws->motionStatus = 1;
|
|
|
|
// Update current player bounding rectangle
|
|
Actor *player = getActor();
|
|
Common::Rect *boundingRect = player->getBoundingRect();
|
|
boundingRect->bottom = (int16)player->getPoint2()->y;
|
|
boundingRect->right = (int16)(player->getPoint2()->x * 2);
|
|
|
|
// Adjust scene bounding rect
|
|
_ws->boundingRect = Common::Rect(195, 115, 445 - boundingRect->right, 345 - boundingRect->bottom);
|
|
|
|
player->show(); // flag |= 1 (show actor)
|
|
player->enable();
|
|
|
|
// Update current player coordinates
|
|
player->getPoint1()->x -= player->getPoint2()->x;
|
|
player->getPoint1()->y -= player->getPoint2()->y;
|
|
|
|
// Update all other actors
|
|
if (_ws->actors.size() > 1) {
|
|
for (uint32 i = 1; i < _ws->actors.size(); i++) {
|
|
Actor *actor = _ws->actors[i];
|
|
|
|
actor->show();
|
|
actor->setDirection(kDirectionNW);
|
|
actor->enable();
|
|
|
|
actor->getPoint1()->x -= actor->getPoint2()->x;
|
|
actor->getPoint1()->y -= actor->getPoint2()->y;
|
|
|
|
actor->getBoundingRect()->bottom = (int16)actor->getPoint2()->y;
|
|
actor->getBoundingRect()->right = (int16)(2 * actor->getPoint2()->x);
|
|
}
|
|
}
|
|
|
|
// Queue scene script
|
|
if (_ws->scriptIndex)
|
|
getScript()->queueScript(_ws->scriptIndex, 0);
|
|
|
|
// Clear the graphic queue (FIXME: not sure why we need to do this again)
|
|
getScreen()->clearGraphicsInQueue();
|
|
|
|
// Load transparency tables
|
|
getScreen()->setupTransTables(3, _ws->cellShadeMask1, _ws->cellShadeMask2, _ws->cellShadeMask3);
|
|
getScreen()->selectTransTable(1);
|
|
|
|
// Setup font
|
|
getText()->loadFont(_ws->font1);
|
|
|
|
// Preload graphics (we are just showing the loading screen
|
|
preload();
|
|
|
|
// Play scene intro dialog
|
|
playIntroSpeech();
|
|
|
|
// Set actor type
|
|
_ws->actorType = actorType[_ws->chapter];
|
|
|
|
// Play intro music
|
|
if (_ws->musicCurrentResourceIndex != kMusicStopped && _ws->chapter != kChapter1)
|
|
getSound()->playMusic(MAKE_RESOURCE(kResourcePackMusic, _ws->musicCurrentResourceIndex));
|
|
else
|
|
getSound()->playMusic(kResourceNone, 0);
|
|
|
|
// Update global values
|
|
_vm->lastScreenUpdate = 1;
|
|
getSharedData()->setFlag(kFlagScene1, true);
|
|
|
|
player->setLastScreenUpdate(_vm->screenUpdateCount);
|
|
player->enable();
|
|
|
|
if (_ws->chapter == kChapter9) {
|
|
changePlayer(1);
|
|
_ws->nextPlayer = kActorInvalid;
|
|
}
|
|
}
|
|
|
|
void Scene::enterLoad() {
|
|
if (!_ws)
|
|
error("[Scene::enterLoad] WorldStats not initialized properly");
|
|
|
|
_vm->setGameFlag(kGameFlagScriptProcessing);
|
|
getScreen()->clearGraphicsInQueue();
|
|
|
|
// Setup scene bounding rect
|
|
_ws->boundingRect.left = 195;
|
|
_ws->boundingRect.top = 115;
|
|
_ws->boundingRect.right = 445 - getActor()->getBoundingRect()->right;
|
|
_ws->boundingRect.bottom = 345 - getActor()->getBoundingRect()->bottom;
|
|
|
|
// Setup transparency table
|
|
getScreen()->setupTransTables(3, _ws->cellShadeMask1, _ws->cellShadeMask2, _ws->cellShadeMask3);
|
|
getScreen()->selectTransTable(1);
|
|
getText()->loadFont(_ws->font1);
|
|
|
|
preload();
|
|
|
|
// Adjust object priority
|
|
if (_ws->objects.size() > 0) {
|
|
int32 priority = 4091;
|
|
|
|
for (uint32 i = 0; i < _ws->objects.size(); i++) {
|
|
Object *object = _ws->objects[i];
|
|
object->setPriority(priority);
|
|
object->flags &= ~kObjectFlagC000;
|
|
priority -= 4;
|
|
}
|
|
}
|
|
|
|
// Play intro music
|
|
if (_ws->musicCurrentResourceIndex != kMusicStopped)
|
|
getSound()->playMusic(MAKE_RESOURCE(kResourcePackMusic, _ws->musicCurrentResourceIndex));
|
|
else
|
|
getSound()->playMusic(kResourceNone, 0);
|
|
|
|
// Palette fade
|
|
getScreen()->paletteFade(0, 75, 8);
|
|
getScreen()->clear();
|
|
|
|
getSharedData()->setFlag(kFlagScene1, true);
|
|
_vm->lastScreenUpdate = 1;
|
|
getActor()->setLastScreenUpdate(_vm->screenUpdateCount);
|
|
}
|
|
|
|
void Scene::load(ResourcePackId packId) {
|
|
// Setup resource manager
|
|
_packId = packId;
|
|
getResource()->setMusicPackId(packId);
|
|
|
|
char filename[10];
|
|
snprintf(filename, 10, SCENE_FILE_MASK, _packId);
|
|
|
|
char sceneTag[6];
|
|
Common::File* fd = new Common::File;
|
|
|
|
if (!Common::File::exists(filename))
|
|
error("Scene file doesn't exist %s", filename);
|
|
|
|
fd->open(filename);
|
|
|
|
if (!fd->isOpen())
|
|
error("Failed to load scene file %s", filename);
|
|
|
|
fd->read(sceneTag, 6);
|
|
|
|
if (Common::String(sceneTag, 6) != "DFISCN")
|
|
error("The file isn't recognized as scene %s", filename);
|
|
|
|
_ws = new WorldStats(_vm);
|
|
_ws->load(fd);
|
|
|
|
if (_vm->checkGameVersion("Demo"))
|
|
fd->seek(0x1D72E, SEEK_SET);
|
|
|
|
_polygons = new Polygons(fd);
|
|
|
|
if (_vm->checkGameVersion("Demo"))
|
|
fd->seek(3 * 4, SEEK_CUR);
|
|
|
|
ScriptManager *script = getScript();
|
|
script->resetAll();
|
|
script->load(fd);
|
|
|
|
fd->close();
|
|
delete fd;
|
|
|
|
getSharedData()->resetAmbientFlags();
|
|
_ws->field_120 = -1;
|
|
|
|
int32 tick = _vm->getTick();
|
|
for (uint32 a = 0; a < _ws->actors.size(); a++)
|
|
_ws->actors[a]->setLastScreenUpdate(tick);
|
|
|
|
getCursor()->show();
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
// Event handling
|
|
//////////////////////////////////////////////////////////////////////////
|
|
bool Scene::handleEvent(const AsylumEvent &evt) {
|
|
switch ((int32)evt.type) {
|
|
default:
|
|
break;
|
|
|
|
case EVENT_ASYLUM_INIT:
|
|
return init();
|
|
|
|
case EVENT_ASYLUM_ACTIVATE:
|
|
case Common::EVENT_RBUTTONUP:
|
|
activate();
|
|
break;
|
|
|
|
case EVENT_ASYLUM_UPDATE:
|
|
return update();
|
|
|
|
case Common::EVENT_CUSTOM_ENGINE_ACTION_START:
|
|
return action((AsylumAction)evt.customType);
|
|
|
|
case Common::EVENT_KEYDOWN:
|
|
if (evt.kbd.flags & Common::KBD_CTRL)
|
|
_isCTRLPressed = true;
|
|
|
|
return key(evt);
|
|
|
|
case Common::EVENT_KEYUP:
|
|
if (!(evt.kbd.flags & Common::KBD_CTRL))
|
|
_isCTRLPressed = false;
|
|
break;
|
|
|
|
case Common::EVENT_LBUTTONDOWN:
|
|
case Common::EVENT_RBUTTONDOWN:
|
|
case Common::EVENT_MBUTTONDOWN:
|
|
return clickDown(evt);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void Scene::activate() {
|
|
Actor *player = getActor();
|
|
|
|
if (player->getStatus() == kActorStatusWalking)
|
|
player->changeStatus(kActorStatusEnabled);
|
|
|
|
if (player->getStatus() == kActorStatusWalking2)
|
|
player->changeStatus(kActorStatusEnabled2);
|
|
}
|
|
|
|
bool Scene::init() {
|
|
if (!_ws)
|
|
error("[Scene::init] WorldStats not initialized properly");
|
|
|
|
if (getSharedData()->getFlag((kFlag3))) { // this flag is set during an encounter
|
|
getSharedData()->setFlag(kFlag3, false);
|
|
|
|
// The original test for flag 1001 but doesn't use the result
|
|
return true;
|
|
}
|
|
|
|
getCursor()->set(_ws->cursorResources[kCursorResourceScrollUp], 0, kCursorAnimationNone);
|
|
_ws->coordinates[0] = -1;
|
|
getScreen()->clear();
|
|
getText()->loadFont(_ws->font1);
|
|
|
|
ResourceId paletteResource = _ws->actions[getActor()->getActionIndex3()]->paletteResourceId;
|
|
if (!paletteResource)
|
|
paletteResource = _ws->currentPaletteId;
|
|
|
|
getScreen()->setPalette(paletteResource);
|
|
getScreen()->setGammaLevel(paletteResource);
|
|
getScreen()->loadGrayPalette();
|
|
getScreen()->setupTransTables(3, _ws->cellShadeMask1, _ws->cellShadeMask2, _ws->cellShadeMask3);
|
|
getScreen()->selectTransTable(1);
|
|
|
|
getCursor()->show();
|
|
|
|
return true;
|
|
}
|
|
|
|
bool Scene::update() {
|
|
if (getEncounter()->shouldEnablePlayer()) {
|
|
getEncounter()->setShouldEnablePlayer(false);
|
|
|
|
// Enable player
|
|
getActor()->changeStatus(kActorStatusEnabled);
|
|
}
|
|
|
|
uint32 ticks = _vm->getTick();
|
|
|
|
if (!getSharedData()->getFlag(kFlagRedraw)) {
|
|
if (updateScreen())
|
|
return true;
|
|
|
|
getSharedData()->setFlag(kFlagRedraw, true);
|
|
}
|
|
|
|
if (ticks > getSharedData()->getNextScreenUpdate()) {
|
|
if (getSharedData()->getFlag(kFlagRedraw)) {
|
|
if (getSharedData()->getMatteBarHeight() <= 0)
|
|
getScreen()->copyBackBufferToScreen();
|
|
else
|
|
getEncounter()->drawScreen();
|
|
|
|
// Original also sets an unused value to 0
|
|
getSharedData()->setEventUpdate(getSharedData()->getEventUpdate() ^ 1);
|
|
|
|
getSharedData()->setFlag(kFlagRedraw, false);
|
|
getSharedData()->setNextScreenUpdate(ticks + 55);
|
|
++_vm->screenUpdateCount;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool Scene::action(AsylumAction a) {
|
|
switch (a) {
|
|
case kAsylumActionShowVersion:
|
|
// TODO show version!
|
|
break;
|
|
|
|
case kAsylumActionQuickLoad:
|
|
if (!_vm->checkGameVersion("Demo"))
|
|
getSaveLoad()->quickLoad();
|
|
break;
|
|
|
|
case kAsylumActionQuickSave:
|
|
if (!_vm->checkGameVersion("Demo"))
|
|
getSaveLoad()->quickSave();
|
|
break;
|
|
|
|
case kAsylumActionSwitchToSarah:
|
|
case kAsylumActionSwitchToGrimwall:
|
|
case kAsylumActionSwitchToOlmec:
|
|
if (getCursor()->isHidden() || _ws->chapter != kChapter9)
|
|
return true;
|
|
|
|
getScript()->queueScript(_ws->actions[_ws->getActionAreaIndexById(2206 + a - kAsylumActionSwitchToSarah)]->scriptIndex,
|
|
getSharedData()->getPlayerIndex());
|
|
break;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool Scene::key(const AsylumEvent &evt) {
|
|
if (!_ws)
|
|
error("[Scene::key] WorldStats not initialized properly");
|
|
|
|
switch (evt.kbd.keycode) {
|
|
default:
|
|
break;
|
|
|
|
case Common::KEYCODE_BACKSPACE:
|
|
// TODO add support for debug commands
|
|
warning("[Scene::key] debug command handling not implemented!");
|
|
break;
|
|
|
|
case Common::KEYCODE_RETURN:
|
|
// TODO add support for debug commands
|
|
warning("[Scene::key] debug command handling not implemented!");
|
|
break;
|
|
|
|
case Common::KEYCODE_ESCAPE:
|
|
// TODO add support for debug commands
|
|
|
|
if (getSpeech()->getSoundResourceId()) {
|
|
getScene()->stopSpeech();
|
|
} else {
|
|
if (getCursor()->isHidden())
|
|
break;
|
|
|
|
if (!_vm->checkGameVersion("Demo"))
|
|
_vm->switchEventHandler(_vm->menu());
|
|
}
|
|
break;
|
|
|
|
case Common::KEYCODE_LEFTBRACKET:
|
|
if (evt.kbd.ascii != 123)
|
|
break;
|
|
// fallthrough
|
|
|
|
case Common::KEYCODE_p:
|
|
case Common::KEYCODE_q:
|
|
case Common::KEYCODE_r:
|
|
case Common::KEYCODE_s:
|
|
case Common::KEYCODE_t:
|
|
case Common::KEYCODE_u:
|
|
case Common::KEYCODE_v:
|
|
case Common::KEYCODE_w:
|
|
case Common::KEYCODE_x:
|
|
case Common::KEYCODE_y:
|
|
case Common::KEYCODE_z:
|
|
if (speak(evt.kbd.keycode)) {
|
|
_vm->lastScreenUpdate = _vm->screenUpdateCount;
|
|
getActor()->setLastScreenUpdate(_vm->screenUpdateCount);
|
|
}
|
|
break;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool Scene::clickDown(const AsylumEvent &evt) {
|
|
_vm->lastScreenUpdate = 0;
|
|
|
|
if (getSharedData()->getFlag(kFlag2)) {
|
|
stopSpeech();
|
|
|
|
return true;
|
|
}
|
|
|
|
Actor *player = getActor();
|
|
switch (evt.type) {
|
|
default:
|
|
break;
|
|
|
|
case Common::EVENT_RBUTTONDOWN:
|
|
if (getSpeech()->getSoundResourceId())
|
|
stopSpeech();
|
|
|
|
if (player->getStatus() == kActorStatusShowingInventory || player->getStatus() == kActorStatus10) {
|
|
player->changeStatus(kActorStatusEnabled);
|
|
getSound()->playSound(MAKE_RESOURCE(kResourcePackSound, 5));
|
|
} else if (player->getStatus() != kActorStatusDisabled) {
|
|
player->changeStatus(kActorStatusWalking);
|
|
}
|
|
break;
|
|
|
|
case Common::EVENT_MBUTTONDOWN:
|
|
if (player->getStatus() != kActorStatusDisabled) {
|
|
if (player->getStatus() == kActorStatusShowingInventory || player->getStatus() == kActorStatus10)
|
|
player->changeStatus(kActorStatusEnabled);
|
|
else
|
|
player->changeStatus(kActorStatusShowingInventory);
|
|
}
|
|
break;
|
|
|
|
case Common::EVENT_LBUTTONDOWN:
|
|
if (getCursor()->getState() & kCursorStateRight)
|
|
break;
|
|
|
|
if (getSpeech()->getSoundResourceId())
|
|
stopSpeech();
|
|
|
|
if (player->getStatus() == kActorStatusDisabled)
|
|
break;
|
|
|
|
if (player->inventory.getSelectedItem()) {
|
|
if (hitTestPlayer()) {
|
|
player->inventory.selectItem(0);
|
|
return true;
|
|
}
|
|
|
|
HitType type = kHitNone;
|
|
int32 res = hitTestScene(type);
|
|
|
|
if (res == -1)
|
|
getSpeech()->playIndexed(2);
|
|
else
|
|
handleHit(res, type);
|
|
|
|
return true;
|
|
}
|
|
|
|
if (!hitTestPlayer() || player->getStatus() >= kActorStatus11 || !player->inventory[0]) {
|
|
if (player->getStatus() == kActorStatusShowingInventory || player->getStatus() == kActorStatus10) {
|
|
clickInventory();
|
|
} else {
|
|
HitType type = kHitNone;
|
|
int32 res = hitTest(type);
|
|
if (res != -1)
|
|
handleHit(res, type);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
if (player->getStatus() == kActorStatusShowingInventory || player->getStatus() == kActorStatus10) {
|
|
getSound()->playSound(MAKE_RESOURCE(kResourcePackSound, 5));
|
|
player->changeStatus(kActorStatusEnabled);
|
|
} else {
|
|
getSound()->playSound(MAKE_RESOURCE(kResourcePackSound, 2));
|
|
player->changeStatus(kActorStatusShowingInventory);
|
|
}
|
|
break;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
// Scene update
|
|
//////////////////////////////////////////////////////////////////////////
|
|
bool Scene::updateScreen() {
|
|
_frameCounter++;
|
|
|
|
if (updateScene())
|
|
return true;
|
|
|
|
if (Config.performance <= 4) {
|
|
// TODO when Config.performance <= 4, we need to skip drawing frames to screen
|
|
|
|
if (drawScene())
|
|
return true;
|
|
|
|
} else {
|
|
if (drawScene())
|
|
return true;
|
|
}
|
|
|
|
getActor()->drawNumber();
|
|
|
|
// Original handle all debug commands here (we do it as part of each update command)
|
|
|
|
if (getSharedData()->getFlag(kFlagScene1)) {
|
|
getScreen()->clear();
|
|
|
|
getScreen()->stopPaletteFade(0, 0, 0);
|
|
updateScene();
|
|
drawScene();
|
|
getScreen()->copyBackBufferToScreen();
|
|
|
|
getScreen()->stopPaletteFadeAndSet(getWorld()->currentPaletteId, 100, 10);
|
|
drawScene();
|
|
getScreen()->copyBackBufferToScreen();
|
|
|
|
getSharedData()->setFlag(kFlagScene1, false);
|
|
}
|
|
|
|
if (getSpeech()->getSoundResourceId() != 0) {
|
|
if (getSound()->isPlaying(getSpeech()->getSoundResourceId())) {
|
|
getSpeech()->prepareSpeech();
|
|
} else {
|
|
getSpeech()->resetResourceIds();
|
|
_vm->clearGameFlag(kGameFlag219);
|
|
}
|
|
}
|
|
|
|
if (getWorld()->chapter == kChapter5) {
|
|
if (_vm->isGameFlagSet(kGameFlag249))
|
|
drawRain();
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
bool Scene::updateScene() {
|
|
#ifdef DEBUG_SCENE_TIMES
|
|
#define MEASURE_TICKS(func) { \
|
|
int32 startTick =_vm->getTick(); \
|
|
func(); \
|
|
debugC(kDebugLevelScene, "[Scene] " #func " - Time: %d", _vm->getTick() - startTick); \
|
|
}
|
|
#else
|
|
#define MEASURE_TICKS(func) func();
|
|
#endif
|
|
|
|
// Update each part of the scene
|
|
if (getSharedData()->getMatteBarHeight() != 170 || getSharedData()->getMattePlaySound()) {
|
|
MEASURE_TICKS(updateMouse);
|
|
MEASURE_TICKS(updateActors);
|
|
MEASURE_TICKS(updateObjects);
|
|
MEASURE_TICKS(updateAmbientSounds);
|
|
MEASURE_TICKS(updateMusic);
|
|
MEASURE_TICKS(updateAdjustScreen);
|
|
}
|
|
|
|
return getScript()->process();
|
|
}
|
|
|
|
void Scene::updateMouse() {
|
|
Actor *player = getActor();
|
|
Common::Point mouse = getCursor()->position();
|
|
|
|
Common::Point pt;
|
|
player->adjustCoordinates(&pt);
|
|
|
|
Common::Rect actorRect;
|
|
if (_ws->chapter != kChapter2 || getSharedData()->getPlayerIndex() != 10) {
|
|
actorRect.left = pt.x + 20;
|
|
actorRect.top = pt.y;
|
|
actorRect.right = (int16)(pt.x + 2 * player->getPoint2()->x);
|
|
actorRect.bottom = (int16)(pt.y + player->getPoint2()->y);
|
|
} else {
|
|
actorRect.left = pt.x + 50;
|
|
actorRect.top = pt.y + 60;
|
|
actorRect.right = (int16)(pt.x + getActor(10)->getPoint2()->x + 10);
|
|
actorRect.bottom = (int16)(pt.y + getActor(10)->getPoint2()->y - 20);
|
|
}
|
|
|
|
ActorDirection newDirection = kDirectionInvalid;
|
|
|
|
if (mouse.x < actorRect.left) {
|
|
if (mouse.y >= actorRect.top) {
|
|
if (mouse.y > actorRect.bottom) {
|
|
if (player->getDirection() == kDirectionW) {
|
|
if ((mouse.y - actorRect.bottom) > 10)
|
|
newDirection = kDirectionSW;
|
|
} else {
|
|
if (player->getDirection() == kDirectionS) {
|
|
if ((actorRect.left - mouse.x) > 10)
|
|
newDirection = kDirectionSW;
|
|
} else {
|
|
newDirection = kDirectionSW;
|
|
}
|
|
}
|
|
} else {
|
|
if (player->getDirection() == kDirectionNW) {
|
|
if ((mouse.y - actorRect.top) > 10)
|
|
newDirection = kDirectionW;
|
|
} else {
|
|
if (player->getDirection() == kDirectionSW) {
|
|
if ((actorRect.bottom - mouse.y) > 10)
|
|
newDirection = kDirectionW;
|
|
} else {
|
|
newDirection = kDirectionW;
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
if (player->getDirection() != kDirectionN) {
|
|
if (player->getDirection() == kDirectionW) {
|
|
if ((actorRect.top - mouse.y) > 10)
|
|
newDirection = kDirectionNW;
|
|
} else {
|
|
newDirection = kDirectionNW;
|
|
}
|
|
} else {
|
|
if ((actorRect.left - mouse.x) > 10)
|
|
newDirection = kDirectionNW;
|
|
}
|
|
}
|
|
|
|
} else if (mouse.x <= actorRect.right) {
|
|
if (mouse.y >= actorRect.top) {
|
|
if (mouse.y > actorRect.bottom) {
|
|
if (player->getDirection() == kDirectionSW) {
|
|
if ((mouse.x - actorRect.left) > 10)
|
|
newDirection = kDirectionS;
|
|
} else {
|
|
if (player->getDirection() == kDirectionSE) {
|
|
if ((actorRect.right - mouse.x) > 10)
|
|
newDirection = kDirectionS;
|
|
} else {
|
|
newDirection = kDirectionS;
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
if (player->getDirection() == kDirectionNW) {
|
|
if ((mouse.x - actorRect.left) > 10)
|
|
newDirection = kDirectionN;
|
|
} else {
|
|
if (player->getDirection() == kDirectionNE) {
|
|
if ((actorRect.right - mouse.x) > 10)
|
|
newDirection = kDirectionN;
|
|
} else {
|
|
newDirection = kDirectionN;
|
|
}
|
|
}
|
|
}
|
|
} else if (mouse.y < actorRect.top) {
|
|
if (player->getDirection() != kDirectionN) {
|
|
if (player->getDirection() == kDirectionE) {
|
|
if ((actorRect.top - mouse.y) > 10)
|
|
newDirection = kDirectionNE;
|
|
} else {
|
|
newDirection = kDirectionNE;
|
|
}
|
|
} else {
|
|
if ((mouse.x - actorRect.right) > 10)
|
|
newDirection = kDirectionNE;
|
|
}
|
|
} else if (mouse.y <= actorRect.bottom) {
|
|
if (player->getDirection() == kDirectionSE) {
|
|
if ((actorRect.bottom - mouse.y) > 10)
|
|
newDirection = kDirectionE;
|
|
} else {
|
|
if (player->getDirection() == kDirectionNE) {
|
|
if ((mouse.y - actorRect.top) > 10)
|
|
newDirection = kDirectionE;
|
|
} else {
|
|
newDirection = kDirectionE;
|
|
}
|
|
}
|
|
} else if (player->getDirection() == kDirectionS) {
|
|
if ((mouse.x - actorRect.right) > 10)
|
|
newDirection = kDirectionSE;
|
|
} else if ((player->getDirection() != kDirectionE || (mouse.y - actorRect.bottom) > 10)) {
|
|
newDirection = kDirectionSE;
|
|
}
|
|
|
|
updateCursor(newDirection, actorRect);
|
|
|
|
if (newDirection >= kDirectionN)
|
|
if (player->getStatus() == kActorStatusWalking || player->getStatus() == kActorStatusWalking2)
|
|
player->changeDirection(newDirection);
|
|
}
|
|
|
|
|
|
void Scene::updateActors() {
|
|
if (!_ws)
|
|
error("[Scene::updateActors] WorldStats not initialized properly!");
|
|
|
|
for (uint32 i = 0; i < _ws->actors.size(); i++)
|
|
_ws->actors[i]->update();
|
|
}
|
|
|
|
void Scene::updateObjects() {
|
|
if (!_ws)
|
|
error("[Scene::updateObjects] WorldStats not initialized properly!");
|
|
|
|
for (uint32 i = 0; i < _ws->objects.size(); i++)
|
|
_ws->objects[i]->update();
|
|
}
|
|
|
|
void Scene::updateAmbientSounds() {
|
|
if (!_ws)
|
|
error("[Scene::updateAmbientSounds] WorldStats not initialized properly!");
|
|
|
|
if (Config.performance <= 3)
|
|
return;
|
|
|
|
// The original loops for each actor, but the volume calculation is always the same
|
|
|
|
for (uint32 i = 0; i < _ws->numAmbientSounds; i++) {
|
|
bool processSound = true;
|
|
AmbientSoundItem *snd = &_ws->ambientSounds[i];
|
|
uint32 ambientTick = getSharedData()->getAmbientTick(i);
|
|
|
|
for (int32 f = 0; f < 6; f++) {
|
|
int32 gameFlag = snd->flagNum[f];
|
|
if (gameFlag == 99999)
|
|
continue;
|
|
|
|
if (gameFlag >= 0) {
|
|
if (_vm->isGameFlagNotSet((GameFlag)gameFlag)) {
|
|
processSound = false;
|
|
break;
|
|
}
|
|
} else {
|
|
if (_vm->isGameFlagSet((GameFlag)-gameFlag)) {
|
|
processSound = false;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (processSound) {
|
|
if (_vm->sound()->isPlaying(snd->resourceId)) {
|
|
|
|
if (snd->field_0) {
|
|
int32 volume = Config.ambientVolume + getSound()->calculateVolumeAdjustement(snd->point, snd->attenuation, snd->delta);
|
|
|
|
if (volume <= 0) {
|
|
if (volume < -10000)
|
|
volume = -10000;
|
|
|
|
getSound()->setVolume(snd->resourceId, volume);
|
|
} else {
|
|
getSound()->setVolume(snd->resourceId, 0);
|
|
}
|
|
}
|
|
|
|
} else {
|
|
int32 panning = (snd->field_0) ? getSound()->calculatePanningAtPoint(snd->point) : 0;
|
|
|
|
int32 volume = 0;
|
|
if (snd->field_0)
|
|
volume = getSound()->calculateVolumeAdjustement(snd->point, snd->attenuation, snd->delta);
|
|
else
|
|
volume = -(int32)pow((double)snd->delta, 2);
|
|
|
|
volume += Config.ambientVolume;
|
|
|
|
|
|
if (LOBYTE(snd->flags) & 1) {
|
|
|
|
getSound()->playSound(snd->resourceId, true, volume, panning);
|
|
|
|
} else if (LOBYTE(snd->flags) & 2) {
|
|
if (_vm->getRandom(10000) < 10) {
|
|
if (snd->field_0) {
|
|
getSound()->playSound(snd->resourceId, false, volume, panning);
|
|
} else {
|
|
int32 tmpVol = volume + (int32)_vm->getRandom(500) * ((_vm->getRandom(100) >= 50) ? -1 : 1);
|
|
|
|
if (tmpVol <= -10000)
|
|
tmpVol = -10000;
|
|
|
|
if (tmpVol >= 0)
|
|
tmpVol = 0;
|
|
else if (tmpVol <= -10000)
|
|
tmpVol = -10000;
|
|
|
|
getSound()->playSound(snd->resourceId, false, tmpVol, _vm->getRandom(20001) - 10000);
|
|
}
|
|
}
|
|
} else if (LOBYTE(snd->flags) & 4) {
|
|
if (ambientTick > _vm->getTick()) {
|
|
if (snd->nextTick >= 0)
|
|
getSharedData()->setAmbientTick(i, (uint32)((int32)_vm->getTick() + snd->nextTick * 60000));
|
|
else
|
|
getSharedData()->setAmbientTick(i, (uint32)((int32)_vm->getTick() - snd->nextTick * 1000));
|
|
|
|
getSound()->playSound(snd->resourceId, false, volume, panning);
|
|
}
|
|
} else if (LOBYTE(snd->flags) & 8) {
|
|
if (!getSharedData()->getAmbientFlag(i)) {
|
|
getSound()->playSound(snd->resourceId, false, volume, panning);
|
|
getSharedData()->setAmbientFlag(i, 1);
|
|
}
|
|
|
|
}
|
|
}
|
|
} else {
|
|
if (_vm->sound()->isPlaying(snd->resourceId))
|
|
_vm->sound()->stop(snd->resourceId);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Scene::updateMusic() {
|
|
if (!getWorld()->musicFlag)
|
|
return;
|
|
|
|
if (getWorld()->musicCurrentResourceIndex != kMusicStopped) {
|
|
switch (getWorld()->musicStatus) {
|
|
default:
|
|
break;
|
|
|
|
case 1:
|
|
if (getWorld()->musicResourceIndex == kMusicStopped) {
|
|
getWorld()->musicCurrentResourceIndex = kMusicStopped;
|
|
getWorld()->musicStatus = 0;
|
|
getSound()->playMusic(kResourceNone, 0);
|
|
} else {
|
|
getWorld()->musicCurrentResourceIndex = getWorld()->musicResourceIndex;
|
|
getWorld()->musicStatus = getWorld()->musicStatusExt;
|
|
getSound()->playMusic(MAKE_RESOURCE(kResourcePackMusic, getWorld()->musicResourceIndex));
|
|
}
|
|
getWorld()->musicResourceIndex = kMusicStopped;
|
|
getWorld()->musicStatusExt = 0;
|
|
getWorld()->musicFlag = 0;
|
|
break;
|
|
|
|
case 2:
|
|
_musicVolume = getSound()->getMusicVolume();
|
|
getWorld()->musicStatus = 4;
|
|
break;
|
|
|
|
case 4:
|
|
_musicVolume -= 150;
|
|
if (_musicVolume > -2500) {
|
|
getSound()->setMusicVolume(_musicVolume);
|
|
break;
|
|
}
|
|
|
|
_musicVolume = -10000;
|
|
getWorld()->musicCurrentResourceIndex = kMusicStopped;
|
|
|
|
if (getWorld()->musicResourceIndex == kMusicStopped) {
|
|
getWorld()->musicStatus = 0;
|
|
getSound()->playMusic(kResourceNone, 0);
|
|
getWorld()->musicResourceIndex = kMusicStopped;
|
|
getWorld()->musicStatusExt = 0;
|
|
getWorld()->musicFlag = 0;
|
|
} else {
|
|
getWorld()->musicStatus = 8;
|
|
getSound()->playMusic(kResourceNone, 0);
|
|
getWorld()->musicCurrentResourceIndex = getWorld()->musicResourceIndex;
|
|
_musicVolume = -2500;
|
|
getSound()->playMusic(MAKE_RESOURCE(kResourcePackMusic, getWorld()->musicResourceIndex), _musicVolume);
|
|
}
|
|
break;
|
|
|
|
case 8:
|
|
_musicVolume += 150;
|
|
if (_musicVolume < Config.musicVolume) {
|
|
getSound()->setMusicVolume(_musicVolume);
|
|
break;
|
|
}
|
|
|
|
getSound()->setMusicVolume(Config.musicVolume);
|
|
getWorld()->musicStatus = getWorld()->musicStatusExt;
|
|
getWorld()->musicResourceIndex = kMusicStopped;
|
|
getWorld()->musicStatusExt = 0;
|
|
getWorld()->musicFlag = 0;
|
|
break;
|
|
}
|
|
} else if (getWorld()->musicResourceIndex != kMusicStopped) {
|
|
switch (getWorld()->musicStatusExt) {
|
|
default:
|
|
break;
|
|
|
|
case 1:
|
|
getWorld()->musicCurrentResourceIndex = getWorld()->musicResourceIndex;
|
|
getWorld()->musicStatus = 1;
|
|
getSound()->playMusic(MAKE_RESOURCE(kResourcePackMusic, getWorld()->musicResourceIndex));
|
|
getWorld()->musicResourceIndex = kMusicStopped;
|
|
getWorld()->musicStatusExt = 0;
|
|
getWorld()->musicFlag = 0;
|
|
break;
|
|
|
|
case 2:
|
|
_musicVolume = -10000;
|
|
getSound()->setMusicVolume(_musicVolume);
|
|
getWorld()->musicCurrentResourceIndex = getWorld()->musicResourceIndex;
|
|
getWorld()->musicStatus = 8;
|
|
getSound()->playMusic(MAKE_RESOURCE(kResourcePackMusic, getWorld()->musicResourceIndex), _musicVolume);
|
|
break;
|
|
}
|
|
} else {
|
|
getWorld()->musicFlag = 0;
|
|
}
|
|
}
|
|
|
|
void Scene::updateAdjustScreen() {
|
|
if (g_debugScrolling) {
|
|
debugScreenScrolling();
|
|
} else {
|
|
updateCoordinates();
|
|
}
|
|
}
|
|
|
|
void Scene::updateCoordinates() {
|
|
Actor *act = getActor();
|
|
int16 xLeft = _ws->xLeft, oxLeft = xLeft;
|
|
int16 yTop = _ws->yTop, oyTop = yTop;
|
|
int16 posX = act->getPoint1()->x - _ws->xLeft;
|
|
int16 posY = act->getPoint1()->y - _ws->yTop;
|
|
Common::Rect boundingRect = _ws->boundingRect;
|
|
|
|
switch (_ws->motionStatus) {
|
|
default:
|
|
break;
|
|
|
|
case 1:
|
|
if (posX < boundingRect.left) {
|
|
xLeft = (posX - boundingRect.left) + _ws->xLeft;
|
|
_ws->xLeft += posX - boundingRect.left;
|
|
} else if (posX > boundingRect.right) {
|
|
xLeft = (posX - boundingRect.right) + _ws->xLeft;
|
|
_ws->xLeft += posX - boundingRect.right;
|
|
}
|
|
|
|
if (posY < boundingRect.top) {
|
|
yTop = (posY - boundingRect.top) + _ws->yTop;
|
|
_ws->yTop += posY - boundingRect.top;
|
|
} else if (posY > boundingRect.bottom) {
|
|
yTop = (posY - boundingRect.bottom) + _ws->yTop;
|
|
_ws->yTop += posY - boundingRect.bottom;
|
|
}
|
|
|
|
if (xLeft < 0)
|
|
xLeft = _ws->xLeft = 0;
|
|
|
|
if (xLeft > (_ws->width - 640))
|
|
xLeft = _ws->xLeft = _ws->width - 640;
|
|
|
|
if (yTop < 0)
|
|
yTop = _ws->yTop = 0;
|
|
|
|
if (yTop > (_ws->height - 480))
|
|
yTop = _ws->yTop = _ws->height - 480;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
case 5: {
|
|
getSharedData()->setSceneOffset(getSharedData()->getSceneOffset() + getSharedData()->getSceneOffsetAdd());
|
|
|
|
int32 coord1 = 0;
|
|
int32 coord2 = 0;
|
|
|
|
if (abs(getSharedData()->getSceneCoords().x - _ws->coordinates[0]) <= abs(getSharedData()->getSceneCoords().y - _ws->coordinates[1])) {
|
|
coord1 = _ws->coordinates[1];
|
|
coord2 = yTop;
|
|
|
|
if (_ws->coordinates[1] != _ws->yTop)
|
|
xLeft = _ws->xLeft = getSharedData()->getSceneOffset() + getSharedData()->getSceneCoords().x;
|
|
|
|
yTop = _ws->coordinates[2] + _ws->yTop;
|
|
_ws->yTop += _ws->coordinates[2];
|
|
} else {
|
|
coord1 = _ws->coordinates[0];
|
|
coord2 = xLeft;
|
|
|
|
if (_ws->coordinates[0] != _ws->xLeft)
|
|
yTop = _ws->yTop = getSharedData()->getSceneOffset() + getSharedData()->getSceneCoords().y;
|
|
|
|
xLeft = _ws->coordinates[2] + _ws->xLeft;
|
|
_ws->xLeft += _ws->coordinates[2];
|
|
}
|
|
|
|
if (abs(coord2 - coord1) <= abs(_ws->coordinates[2])) {
|
|
_ws->motionStatus = 3;
|
|
_ws->coordinates[0] = -1;
|
|
}
|
|
|
|
}
|
|
break;
|
|
}
|
|
|
|
// Update scene coordinates
|
|
Common::Rect sceneRect = _ws->sceneRects[_ws->sceneRectIdx];
|
|
|
|
if (xLeft < sceneRect.left)
|
|
xLeft =_ws->xLeft =sceneRect.left;
|
|
|
|
if (yTop < sceneRect.top)
|
|
yTop = _ws->yTop = sceneRect.top;
|
|
|
|
if ((xLeft + 639) > sceneRect.right)
|
|
xLeft = _ws->xLeft = sceneRect.right - 639;
|
|
|
|
if ((yTop + 479) > sceneRect.bottom)
|
|
yTop = _ws->yTop = sceneRect.bottom - 479;
|
|
|
|
// XXX dword_44E1EC is set to 2 at this point if the scene coordinates
|
|
// have changed, but that variable is never used anywhere else
|
|
if ((_ws->motionStatus == 2 || _ws->motionStatus == 5) && (oxLeft != _ws->xLeft || oyTop != _ws->yTop))
|
|
debugC(kDebugLevelScene, "[Scene::updateCoordinates] (%d, %d) ~> (%d, %d), motionStatus = %d",
|
|
_ws->xLeft, _ws->yTop, _ws->coordinates[0], _ws->coordinates[1], _ws->motionStatus);
|
|
}
|
|
|
|
void Scene::updateCursor(ActorDirection direction, const Common::Rect &rect) {
|
|
HitType type = kHitNone;
|
|
Actor *player = getActor();
|
|
int16 rightLimit = rect.right - 10;
|
|
Common::Point mouse = getCursor()->position();
|
|
|
|
if (getSharedData()->getFlag(kFlagIsEncounterRunning)) {
|
|
if (getCursor()->getResourceId() != _ws->cursorResources[kCursorResourceTalkNPC])
|
|
getCursor()->set(_ws->cursorResources[kCursorResourceTalkNPC]);
|
|
|
|
return;
|
|
}
|
|
|
|
if (getCursor()->getState() & kCursorStateRight) {
|
|
if (player->getStatus() == kActorStatusWalking || player->getStatus() == kActorStatusWalking2) {
|
|
|
|
ResourceId resourceId =_ws->cursorResources[direction];
|
|
|
|
if (direction >= kDirectionN && getCursor()->getResourceId() != resourceId)
|
|
getCursor()->set(resourceId);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
if (player->getStatus() == kActorStatusShowingInventory || player->getStatus() == kActorStatus10) {
|
|
if (getCursor()->getResourceId() != _ws->cursorResources[kCursorResourceHand])
|
|
getCursor()->set(_ws->cursorResources[kCursorResourceHand]);
|
|
|
|
return;
|
|
}
|
|
|
|
if (player->inventory.getSelectedItem()) {
|
|
if (mouse.x >= rect.left && mouse.x <= rightLimit && mouse.y >= rect.top && mouse.y <= rect.bottom && hitTestPlayer()) {
|
|
|
|
ResourceId id = _ws->inventoryCursorsNormal[player->inventory.getSelectedItem() - 1];
|
|
if (getCursor()->getResourceId() != id)
|
|
getCursor()->set(id, 0, kCursorAnimationNone);
|
|
|
|
} else {
|
|
if (hitTestScene(type) == -1) {
|
|
ResourceId id = _ws->inventoryCursorsNormal[player->inventory.getSelectedItem() - 1];
|
|
if (getCursor()->getResourceId() != id)
|
|
getCursor()->set(id, 0, kCursorAnimationNone);
|
|
} else {
|
|
ResourceId id = _ws->inventoryCursorsBlinking[player->inventory.getSelectedItem() - 1];
|
|
uint32 frameCount = GraphicResource::getFrameCount(_vm, id);
|
|
if (getCursor()->getResourceId() != id)
|
|
getCursor()->set(id, 0, (frameCount <= 1) ? kCursorAnimationNone : kCursorAnimationMirror);
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
if (mouse.x >= rect.left && mouse.x <= rightLimit && mouse.y >= rect.top && mouse.y <= rect.bottom && hitTestPlayer()) {
|
|
if (player->inventory[0]) {
|
|
if (getCursor()->getResourceId() != _ws->cursorResources[kCursorResourceGrabPointer])
|
|
getCursor()->set(_ws->cursorResources[kCursorResourceGrabPointer]);
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
int32 index = hitTest(type);
|
|
if (index == -1) {
|
|
if (_ws->chapter != kChapter2 || getSharedData()->getPlayerIndex() != 10) {
|
|
if (getCursor()->getResourceId() != _ws->cursorResources[kCursorResourceMagnifyingGlass] || getCursor()->getAnimation())
|
|
getCursor()->set(_ws->cursorResources[kCursorResourceMagnifyingGlass]);
|
|
} else {
|
|
if (getCursor()->getResourceId() != _ws->cursorResources[kCursorResourceTalkNPC2] || getCursor()->getAnimation())
|
|
getCursor()->set(_ws->cursorResources[kCursorResourceTalkNPC2]);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
int32 actionType = 0;
|
|
switch (type) {
|
|
default:
|
|
error("[Scene::updateCursor] Invalid hit type!");
|
|
break;
|
|
|
|
case kHitActionArea:
|
|
actionType = _ws->actions[index]->actionType;
|
|
break;
|
|
|
|
case kHitObject:
|
|
actionType = _ws->objects[index]->actionType;
|
|
break;
|
|
|
|
case kHitActor:
|
|
actionType = getActor(index)->actionType;
|
|
break;
|
|
}
|
|
|
|
if (actionType & kActionTypeFind) {
|
|
if (getCursor()->getResourceId() != _ws->cursorResources[kCursorResourceMagnifyingGlass] || getCursor()->getAnimation() != kCursorAnimationMirror)
|
|
getCursor()->set(_ws->cursorResources[kCursorResourceMagnifyingGlass]);
|
|
} else if (actionType & kActionTypeTalk) {
|
|
if (getCursor()->getResourceId() != _ws->cursorResources[kCursorResourceTalkNPC])
|
|
getCursor()->set(_ws->cursorResources[kCursorResourceTalkNPC]);
|
|
} else if (actionType & kActionTypeGrab) {
|
|
if (getCursor()->getResourceId() != _ws->cursorResources[kCursorResourceHand])
|
|
getCursor()->set(_ws->cursorResources[kCursorResourceHand]);
|
|
} else if (actionType & kActionType16) {
|
|
if (getCursor()->getResourceId() != _ws->cursorResources[kCursorResourceTalkNPC2] || getCursor()->getAnimation() != kCursorAnimationMirror)
|
|
getCursor()->set(_ws->cursorResources[kCursorResourceTalkNPC2]);
|
|
} else if (_ws->chapter != kChapter2 && getSharedData()->getPlayerIndex() != 10) {
|
|
if (getCursor()->getResourceId() != _ws->cursorResources[kCursorResourceMagnifyingGlass] || getCursor()->getAnimation())
|
|
getCursor()->set(_ws->cursorResources[kCursorResourceMagnifyingGlass]);
|
|
} else {
|
|
if (getCursor()->getResourceId() != _ws->cursorResources[kCursorResourceTalkNPC2] || getCursor()->getAnimation())
|
|
getCursor()->set(_ws->cursorResources[kCursorResourceTalkNPC2]);
|
|
}
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
// HitTest
|
|
//////////////////////////////////////////////////////////////////////////
|
|
int32 Scene::hitTest(HitType &type) {
|
|
type = kHitNone;
|
|
|
|
int32 targetIdx = hitTestObject();
|
|
if (targetIdx == -1) {
|
|
|
|
targetIdx = hitTestActionArea();
|
|
if (targetIdx == -1) {
|
|
|
|
targetIdx = hitTestActor();
|
|
type = kHitActor;
|
|
|
|
} else {
|
|
type = kHitActionArea;
|
|
}
|
|
} else {
|
|
type = kHitObject;
|
|
}
|
|
|
|
return targetIdx;
|
|
}
|
|
|
|
int32 Scene::hitTestScene(HitType &type) {
|
|
if (!_ws)
|
|
error("[Scene::hitTestScene] WorldStats not initialized properly!");
|
|
|
|
const Common::Point pt = getCursor()->position();
|
|
|
|
int16 top = pt.x + _ws->xLeft;
|
|
int16 left = pt.y + _ws->yTop;
|
|
type = kHitNone;
|
|
|
|
int32 index = findActionArea(kActionAreaType2, Common::Point(top, left));
|
|
if (index != -1) {
|
|
if (_ws->actions[index]->actionType & kActionType8) {
|
|
type = kHitActionArea;
|
|
return index;
|
|
}
|
|
}
|
|
|
|
// Check objects
|
|
for (uint i = 0; i < _ws->objects.size(); i++) {
|
|
Object *object = _ws->objects[i];
|
|
|
|
if (object->isOnScreen() && (object->actionType & kActionType8)) {
|
|
if (hitTestPixel(object->getResourceId(),
|
|
object->getFrameIndex(),
|
|
top - object->x,
|
|
left - object->y,
|
|
(bool)(object->flags & kObjectFlag1000))) {
|
|
type = kHitObject;
|
|
return i;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Check actors
|
|
for (uint i = 0; i < _ws->actors.size(); i++) {
|
|
Actor *actor = _ws->actors[i];
|
|
|
|
if (actor->actionType & kActionType8) {
|
|
uint32 frameIndex = (actor->getFrameIndex() >= actor->getFrameCount() ? 2 * actor->getFrameCount() - (actor->getFrameIndex() + 1) : actor->getFrameIndex());
|
|
|
|
if (hitTestPixel(actor->getResourceId(),
|
|
frameIndex,
|
|
top - (actor->getPoint()->x + actor->getPoint1()->x),
|
|
left - (actor->getPoint()->y + actor->getPoint1()->y),
|
|
actor->getDirection() >= kDirectionSE)) {
|
|
type = kHitActor;
|
|
return i;
|
|
}
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
int32 Scene::hitTestActionArea() {
|
|
const Common::Point pt = getCursor()->position();
|
|
|
|
int32 targetIdx = findActionArea(kActionAreaType2, Common::Point(_ws->xLeft + pt.x, _ws->yTop + pt.y));
|
|
|
|
if (targetIdx == -1 || !(_ws->actions[targetIdx]->actionType & (kActionTypeFind | kActionTypeTalk | kActionTypeGrab | kActionType16)))
|
|
return -1;
|
|
|
|
return targetIdx;
|
|
}
|
|
|
|
ActorIndex Scene::hitTestActor() {
|
|
if (!_ws)
|
|
error("[Scene::hitTestActor] WorldStats not initialized properly!");
|
|
|
|
const Common::Point mouse = getCursor()->position();
|
|
|
|
if (_ws->actors.size() == 0)
|
|
return -1;
|
|
|
|
// Check actors 13 to 20
|
|
if (_ws->actors.size() >= 20) {
|
|
for (uint i = 13; i < 21; i++) {
|
|
Actor *actor = getActor(i);
|
|
|
|
if (!actor->isOnScreen() || !actor->actionType)
|
|
continue;
|
|
|
|
Common::Rect rect = GraphicResource::getFrameRect(_vm, getActor(12)->getResourceId(), 0);
|
|
|
|
int32 x = _ws->xLeft + mouse.x - (actor->getPoint1()->x + actor->getPoint()->x);
|
|
int32 y = _ws->yTop + mouse.y - (actor->getPoint1()->y + actor->getPoint()->y);
|
|
|
|
if (x > (rect.left - 20)
|
|
&& x < (rect.width() + rect.left + 20)
|
|
&& y > (rect.top - 20)
|
|
&& y < (rect.height() + rect.top + 20))
|
|
return i;
|
|
}
|
|
}
|
|
|
|
// Check Actor 11
|
|
if (_ws->actors.size() >= 11) {
|
|
Actor *actor11 = getActor(11);
|
|
if (actor11->isOnScreen() && actor11->actionType) {
|
|
Common::Point pt = mouse + Common::Point(_ws->xLeft, _ws->yTop) - *actor11->getPoint1();
|
|
|
|
if (actor11->getBoundingRect()->contains(pt))
|
|
return 11;
|
|
}
|
|
}
|
|
|
|
switch (_ws->chapter) {
|
|
default:
|
|
break;
|
|
|
|
case kChapter8:
|
|
if (_ws->actors.size() < 7)
|
|
error("[Scene::hitTestActor] Not enough actors to check (chapter 8 - checking actors 1-6)!");
|
|
|
|
for (uint i = 1; i < 7; i++) {
|
|
Actor *actor = getActor(i);
|
|
|
|
if (!actor->isVisible() || !actor->actionType)
|
|
continue;
|
|
|
|
int32 x = _ws->xLeft + mouse.x - (actor->getPoint1()->x + actor->getPoint()->x);
|
|
int32 y = _ws->yTop + mouse.y - (actor->getPoint1()->y + actor->getPoint()->y);
|
|
|
|
if (x > 300 && x < 340 && y > 220 && y < 260)
|
|
return i;
|
|
}
|
|
break;
|
|
|
|
case kChapter11:
|
|
if (_ws->actors.size() <= 1)
|
|
error("[Scene::hitTestActor] Not enough actors to check (chapter 11 - checking actor 1)!");
|
|
|
|
if (getActor(1)->isOnScreen() && getActor(1)->actionType) {
|
|
Actor *actor = getActor(1);
|
|
|
|
int32 x = _ws->xLeft + mouse.x - (actor->getPoint1()->x + actor->getPoint()->x);
|
|
int32 y = _ws->yTop + mouse.y - (actor->getPoint1()->y + actor->getPoint()->y);
|
|
|
|
Common::Rect rect = GraphicResource::getFrameRect(_vm, actor->getResourceId(), 0);
|
|
|
|
if (x > (rect.left - 10)
|
|
&& x < (rect.width() + rect.left + 10)
|
|
&& y > (rect.top - 10)
|
|
&& y < (rect.height() + rect.top + 10))
|
|
return 1;
|
|
}
|
|
|
|
if (_ws->actors.size() <= 15)
|
|
error("[Scene::hitTestActor] Not enough actors to check (chapter 11 - checking actors 10-15)!");
|
|
|
|
for (uint i = 10; i < 15; i++) {
|
|
Actor *actor = getActor(i);
|
|
|
|
if (!actor->isOnScreen() || !actor->actionType)
|
|
continue;
|
|
|
|
Common::Rect rect = GraphicResource::getFrameRect(_vm, actor->getResourceId(), 0);
|
|
|
|
int32 x = _ws->xLeft + mouse.x - (actor->getPoint1()->x + actor->getPoint()->x);
|
|
int32 y = _ws->yTop + mouse.y - (actor->getPoint1()->y + actor->getPoint()->y);
|
|
|
|
if (x > (rect.left - 10)
|
|
&& x < (rect.width() + rect.left + 10)
|
|
&& y > (rect.top - 10)
|
|
&& y < (rect.height() + rect.top + 10))
|
|
return i;
|
|
}
|
|
break;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
// Default check
|
|
for (int i = _ws->actors.size() - 1; i >= 0 ; i--) {
|
|
Actor *actor = getActor(i);
|
|
|
|
if (!actor->isOnScreen() || !actor->actionType)
|
|
continue;
|
|
|
|
uint32 hitFrame;
|
|
if (actor->getFrameIndex() >= actor->getFrameCount())
|
|
hitFrame = 2 * actor->getFrameIndex() - (actor->getFrameCount() + 1);
|
|
else
|
|
hitFrame = actor->getFrameIndex();
|
|
|
|
if (hitTestPixel(actor->getResourceId(),
|
|
hitFrame,
|
|
_ws->xLeft + mouse.x - (actor->getPoint1()->x + actor->getPoint()->x),
|
|
_ws->yTop + mouse.y - (actor->getPoint1()->y + actor->getPoint()->y),
|
|
actor->getDirection() >= kDirectionSE))
|
|
return i;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
bool Scene::hitTestPlayer() {
|
|
const Common::Point pt = getCursor()->position();
|
|
|
|
Actor *player = getActor();
|
|
Common::Point point;
|
|
|
|
player->adjustCoordinates(&point);
|
|
|
|
uint32 frameIndex = (player->getFrameIndex() >= player->getFrameCount() ? 2 * player->getFrameCount() - (player->getFrameIndex() + 1) : player->getFrameIndex());
|
|
|
|
return hitTestPixel(player->getResourceId(),
|
|
frameIndex,
|
|
pt.x - (player->getPoint()->x + point.x),
|
|
pt.y - (player->getPoint()->y + point.y),
|
|
player->getDirection() >= kDirectionSE);
|
|
}
|
|
|
|
int32 Scene::hitTestObject() {
|
|
if (!_ws)
|
|
error("[Scene::hitTestObject] WorldStats not initialized properly!");
|
|
|
|
const Common::Point pt = getCursor()->position();
|
|
|
|
for (int32 i = _ws->objects.size() - 1; i >= 0; i--) {
|
|
Object *object = _ws->objects[i];
|
|
if (object->isOnScreen() && object->actionType)
|
|
if (hitTestPixel(object->getResourceId(),
|
|
object->getFrameIndex(),
|
|
_ws->xLeft + pt.x - object->x,
|
|
_ws->yTop + pt.y - object->y,
|
|
(bool)(object->flags & kObjectFlag1000)))
|
|
return i;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
bool Scene::hitTestPixel(ResourceId resourceId, uint32 frameIndex, int16 x, int16 y, bool flipped) {
|
|
if (x < 0 || y < 0)
|
|
return false;
|
|
|
|
GraphicResource *resource = new GraphicResource(_vm, resourceId);
|
|
GraphicFrame *frame = resource->getFrame(frameIndex);
|
|
Common::Rect frameRect = frame->getRect();
|
|
|
|
// Check y coordinates
|
|
if (y < frameRect.top || y >= frameRect.bottom)
|
|
goto cleanup;
|
|
|
|
// Compute left/right x coordinates, using flipped value
|
|
int32 left, right;
|
|
if (flipped) {
|
|
if (getScreen()->getFlag() == -1) {
|
|
left = resource->getData().maxWidth - frameRect.right;
|
|
right = resource->getData().maxWidth - frameRect.left;
|
|
} else {
|
|
left = frameRect.left + 2 * (getScreen()->getFlag() - (frameRect.right / 2));
|
|
right = x;
|
|
}
|
|
} else {
|
|
left = frameRect.left;
|
|
right = frameRect.right;
|
|
}
|
|
|
|
// Check x coordinates
|
|
if (x < left || x >= right)
|
|
goto cleanup;
|
|
|
|
// Check pixel value
|
|
byte *pixel;
|
|
if (flipped) {
|
|
pixel = (byte *)frame->surface.getBasePtr((left - x) + frame->getWidth() - 1, y - frame->y);
|
|
} else {
|
|
pixel = (byte *)frame->surface.getBasePtr(x - left, y - frame->y);
|
|
}
|
|
|
|
if (*pixel == 0)
|
|
goto cleanup;
|
|
|
|
delete resource;
|
|
|
|
return true;
|
|
|
|
cleanup:
|
|
delete resource;
|
|
|
|
return false;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
// Hit actions
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void Scene::handleHit(int32 index, HitType type) {
|
|
if (!_ws)
|
|
error("[Scene::handleHit] WorldStats not initialized properly!");
|
|
|
|
switch (type) {
|
|
default:
|
|
break;
|
|
|
|
case kHitActionArea:
|
|
if (!getScript()->isInQueue(_ws->actions[index]->scriptIndex)) {
|
|
debugC(kDebugLevelScripts, "[Script] Queuing Script idx: %d from kHitActionArea (idx: %d, name: '%s')",
|
|
_ws->actions[index]->scriptIndex, index, _ws->actions[index]->name);
|
|
getScript()->queueScript(_ws->actions[index]->scriptIndex, getSharedData()->getPlayerIndex());
|
|
}
|
|
|
|
switch (_ws->chapter) {
|
|
default:
|
|
break;
|
|
|
|
case kChapter2:
|
|
hitAreaChapter2(_ws->actions[index]->id);
|
|
break;
|
|
|
|
case kChapter7:
|
|
hitAreaChapter7(_ws->actions[index]->id);
|
|
break;
|
|
|
|
case kChapter11:
|
|
hitAreaChapter11(_ws->actions[index]->id);
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case kHitObject: {
|
|
Object *object = _ws->objects[index];
|
|
|
|
if (object->getSoundResourceId()) {
|
|
if (getSound()->isPlaying(object->getSoundResourceId())) {
|
|
getSound()->stop(object->getSoundResourceId());
|
|
object->setSoundResourceId(kResourceNone);
|
|
}
|
|
}
|
|
|
|
if (!getScript()->isInQueue(object->getScriptIndex())) {
|
|
debugC(kDebugLevelScripts, "[Script] Queuing Script idx: %d from kHitObject (id: %d, name: '%s')",
|
|
object->getScriptIndex(), object->getId(), object->getName());
|
|
|
|
getScript()->queueScript(object->getScriptIndex(), getSharedData()->getPlayerIndex());
|
|
}
|
|
|
|
// Original executes special script hit functions, but since there is none defined, we can skip this part
|
|
}
|
|
break;
|
|
|
|
case kHitActor: {
|
|
Actor *actor = _ws->actors[index];
|
|
|
|
if (actor->actionType & (kActionTypeFind | kActionType16)) {
|
|
|
|
if (!getScript()->isInQueue(actor->getScriptIndex())) {
|
|
debugC(kDebugLevelScripts, "[Script] Queuing Script idx: %d from kHitActor (id: %d, name: '%s')",
|
|
actor->getScriptIndex(), index, actor->getName());
|
|
getScript()->queueScript(actor->getScriptIndex(), getSharedData()->getPlayerIndex());
|
|
}
|
|
|
|
} else if (actor->actionType & kActionTypeTalk) {
|
|
|
|
if (getSound()->isPlaying(actor->getSoundResourceId())) {
|
|
if (actor->getStatus() != kActorStatusEnabled)
|
|
actor->changeStatus(kActorStatusEnabled);
|
|
|
|
getSound()->stop(actor->getSoundResourceId());
|
|
actor->setSoundResourceId(kResourceNone);
|
|
}
|
|
|
|
if (!getScript()->isInQueue(actor->getScriptIndex())) {
|
|
debugC(kDebugLevelScripts, "[Script] Queuing Script idx: %d from kActionTypeTalk (actor idx: %d)",
|
|
actor->getScriptIndex(), getSharedData()->getPlayerIndex());
|
|
getScript()->queueScript(actor->getScriptIndex(), getSharedData()->getPlayerIndex());
|
|
}
|
|
}
|
|
|
|
switch (_ws->chapter) {
|
|
default:
|
|
break;
|
|
|
|
case kChapter2:
|
|
hitActorChapter2(index);
|
|
break;
|
|
|
|
case kChapter11:
|
|
hitActorChapter11(index);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
void Scene::clickInventory() {
|
|
const Common::Point mouse = getCursor()->position();
|
|
Common::Point point;
|
|
Actor *player = getActor();
|
|
|
|
player->adjustCoordinates(&point);
|
|
|
|
uint count = player->inventory.find();
|
|
|
|
player->inventory.selectItem(0);
|
|
|
|
if (count > 0) {
|
|
for (uint32 i = 0; i < count; i++) {
|
|
Common::Point ringPoint = Inventory::getInventoryRingPoint(_vm, count, i);
|
|
int32 x = point.x + player->getPoint2()->x + ringPoint.x;
|
|
int32 y = point.y + player->getPoint2()->y / 2 - ringPoint.y;
|
|
|
|
if (mouse.x >= x && mouse.x <= (x + 40) && mouse.y >= y && mouse.y <= (y + 40)) {
|
|
getSound()->playSound(MAKE_RESOURCE(kResourcePackSound, 4));
|
|
|
|
if (_ws->chapter == kChapter9) {
|
|
switch (i) {
|
|
default:
|
|
player->inventory.selectItem(player->inventory[i]);
|
|
break;
|
|
|
|
case 0:
|
|
getScript()->queueScript(_ws->actions[_ws->getActionAreaIndexById(2206)]->scriptIndex, getSharedData()->getPlayerIndex());
|
|
break;
|
|
|
|
case 1:
|
|
getScript()->queueScript(_ws->actions[_ws->getActionAreaIndexById(2207)]->scriptIndex, getSharedData()->getPlayerIndex());
|
|
break;
|
|
|
|
case 2:
|
|
getScript()->queueScript(_ws->actions[_ws->getActionAreaIndexById(2208)]->scriptIndex, getSharedData()->getPlayerIndex());
|
|
break;
|
|
}
|
|
} else {
|
|
player->inventory.selectItem(player->inventory[i]);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
player->changeStatus(kActorStatusEnabled);
|
|
getSound()->playSound(MAKE_RESOURCE(kResourcePackSound, 5));
|
|
}
|
|
|
|
void Scene::hitAreaChapter2(int32 id) {
|
|
if (id == 783)
|
|
getActor()->inventory.selectItem(6);
|
|
}
|
|
|
|
void Scene::hitAreaChapter7(int32 id) {
|
|
switch (id) {
|
|
default:
|
|
break;
|
|
|
|
case 1088:
|
|
if (_isCTRLPressed)
|
|
_vm->setGameFlag(kGameFlag1144);
|
|
break;
|
|
|
|
case 2504:
|
|
if (++_hitAreaChapter7Counter > 20) {
|
|
_vm->setGameFlag(kGameFlag1108);
|
|
getActor(1)->setPosition(570, 225, kDirectionN, 0);
|
|
getActor(1)->show();
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
void Scene::hitAreaChapter11(int32 id) {
|
|
if (!_ws)
|
|
error("[Scene::hitAreaChapter11] WorldStats not initialized properly!");
|
|
|
|
if (id == 1670)
|
|
_ws->field_E849C = 666;
|
|
}
|
|
|
|
|
|
void Scene::hitActorChapter2(ActorIndex index) {
|
|
if (!_ws)
|
|
error("[Scene::hitActorChapter2] WorldStats not initialized properly!");
|
|
|
|
Actor *player = getActor();
|
|
|
|
if (player->getStatus() != kActorStatusEnabled2 && player->getStatus() != kActorStatusWalking2)
|
|
return;
|
|
|
|
if (index == 11) {
|
|
player->faceTarget((uint32)index, kDirectionFromActor);
|
|
player->changeStatus(kActorStatusAttacking);
|
|
|
|
Actor *actor11 = getActor(index);
|
|
|
|
Common::Point pointPlayer(player->getPoint1()->x + player->getPoint2()->x, player->getPoint1()->y + player->getPoint2()->y);
|
|
Common::Point pointActor11(actor11->getPoint1()->x + actor11->getPoint2()->x, actor11->getPoint1()->y + actor11->getPoint2()->y);
|
|
|
|
if (Actor::euclidianDistance(pointPlayer, pointActor11) < 150) {
|
|
if (actor11->getStatus() == kActorStatusWalking2)
|
|
actor11->changeStatus(kActorStatus18);
|
|
|
|
if (actor11->getStatus() == kActorStatusEnabled)
|
|
actor11->changeStatus(kActorStatusEnabled2);
|
|
}
|
|
|
|
getSharedData()->setChapter2ActorIndex(index);
|
|
|
|
} else if (index > 12) {
|
|
player->faceTarget((uint32)(index + 9), kDirectionFromActor);
|
|
player->changeStatus(kActorStatusAttacking);
|
|
getSharedData()->setChapter2ActorIndex(index);
|
|
}
|
|
}
|
|
|
|
void Scene::hitActorChapter11(ActorIndex index) {
|
|
if (!_ws)
|
|
error("[Scene::hitActorChapter11] WorldStats not initialized properly!");
|
|
|
|
if (_ws->field_E848C < 3)
|
|
_ws->field_E849C = index;
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
// Helpers
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void Scene::playIntroSpeech() {
|
|
ResourceId resourceId;
|
|
|
|
switch (_packId) {
|
|
default:
|
|
resourceId = (ResourceId)_packId;
|
|
break;
|
|
|
|
case kResourcePackCourtyardAndChapel:
|
|
resourceId = getSpeech()->playScene(4, 3);
|
|
break;
|
|
|
|
case kResourcePackCave:
|
|
resourceId = getSpeech()->playScene(4, 6);
|
|
break;
|
|
|
|
case kResourcePackLaboratory:
|
|
resourceId = getSpeech()->playScene(4, 7);
|
|
break;
|
|
}
|
|
|
|
getScreen()->clear();
|
|
getScreen()->stopPaletteFade(0, 0, 0);
|
|
|
|
do {
|
|
// Poll events (this ensure we don't freeze the screen)
|
|
Common::Event ev;
|
|
_vm->getEventManager()->pollEvent(ev);
|
|
|
|
g_system->updateScreen();
|
|
g_system->delayMillis(100);
|
|
|
|
} while (getSound()->isPlaying(resourceId));
|
|
}
|
|
|
|
void Scene::stopSpeech() {
|
|
if (_vm->isGameFlagNotSet(kGameFlag219)) {
|
|
if (getSpeech()->getSoundResourceId() != kResourceNone && getSound()->isPlaying(getSpeech()->getSoundResourceId()))
|
|
getSound()->stopAll(getSpeech()->getSoundResourceId());
|
|
else if (getSpeech()->getTick())
|
|
getSpeech()->setTick(_vm->getTick());
|
|
}
|
|
}
|
|
|
|
bool Scene::speak(Common::KeyCode code) {
|
|
if (!_ws)
|
|
error("[Scene::speak] WorldStats not initialized properly!");
|
|
|
|
#define GET_INDEX() ((int)abs((double)_vm->getRandom(RAND_MAX)) & 1)
|
|
|
|
int32 index = -1;
|
|
|
|
switch (code) {
|
|
default:
|
|
break;
|
|
|
|
case Common::KEYCODE_p:
|
|
switch (_ws->actorType) {
|
|
default:
|
|
break;
|
|
|
|
case 0:
|
|
index = GET_INDEX();
|
|
break;
|
|
|
|
case 1:
|
|
case 2:
|
|
case 3:
|
|
index = 1;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case Common::KEYCODE_q:
|
|
switch (_ws->actorType) {
|
|
default:
|
|
break;
|
|
|
|
case 0:
|
|
index = 3 - GET_INDEX();
|
|
break;
|
|
|
|
case 1:
|
|
case 2:
|
|
case 3:
|
|
index = 2;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case Common::KEYCODE_r:
|
|
switch (_ws->actorType) {
|
|
default:
|
|
break;
|
|
|
|
case 0:
|
|
index = 2;
|
|
break;
|
|
|
|
case 1:
|
|
case 2:
|
|
case 3:
|
|
index = 4;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case Common::KEYCODE_s:
|
|
switch (_ws->actorType) {
|
|
default:
|
|
break;
|
|
|
|
case 0:
|
|
index = 5;
|
|
break;
|
|
|
|
case 1:
|
|
case 2:
|
|
case 3:
|
|
index = 3;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case Common::KEYCODE_t:
|
|
switch (_ws->actorType) {
|
|
default:
|
|
break;
|
|
|
|
case 0:
|
|
index = 6;
|
|
break;
|
|
|
|
case 1:
|
|
case 2:
|
|
case 3:
|
|
index = 4;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case Common::KEYCODE_u:
|
|
switch (_ws->actorType) {
|
|
default:
|
|
break;
|
|
|
|
case 0:
|
|
index = 7;
|
|
break;
|
|
|
|
case 1:
|
|
case 2:
|
|
index = 5;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case Common::KEYCODE_v:
|
|
switch (_ws->actorType) {
|
|
default:
|
|
break;
|
|
|
|
case 0:
|
|
index = 8;
|
|
break;
|
|
|
|
case 1:
|
|
case 2:
|
|
index = 6;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case Common::KEYCODE_w:
|
|
switch (_ws->actorType) {
|
|
default:
|
|
break;
|
|
|
|
case 0:
|
|
index = 9;
|
|
break;
|
|
|
|
case 1:
|
|
case 2:
|
|
index = 7;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case Common::KEYCODE_x:
|
|
switch (_ws->actorType) {
|
|
default:
|
|
break;
|
|
|
|
case 0:
|
|
index = 10;
|
|
break;
|
|
|
|
case 1:
|
|
case 2:
|
|
index = 8;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case Common::KEYCODE_y:
|
|
switch (_ws->actorType) {
|
|
default:
|
|
break;
|
|
|
|
case 0:
|
|
index = 11;
|
|
break;
|
|
|
|
case 1:
|
|
case 2:
|
|
index = 9;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case Common::KEYCODE_z:
|
|
switch (_ws->actorType) {
|
|
default:
|
|
break;
|
|
|
|
case 0:
|
|
index = 13 - GET_INDEX();
|
|
break;
|
|
|
|
case 1:
|
|
case 2:
|
|
index = 10;
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case Common::KEYCODE_LEFTBRACKET:
|
|
switch (_ws->actorType) {
|
|
default:
|
|
break;
|
|
|
|
case 0:
|
|
index = 15 - GET_INDEX();
|
|
break;
|
|
|
|
case 2:
|
|
index = 12 - GET_INDEX();
|
|
break;
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (getSpeech()->getSoundResourceId() && getSound()->isPlaying(getSpeech()->getSoundResourceId()))
|
|
return false;
|
|
|
|
if (index == -1)
|
|
return false;
|
|
|
|
getSpeech()->playPlayer(index);
|
|
|
|
return true;
|
|
|
|
#undef GET_INDEX
|
|
}
|
|
|
|
bool Scene::pointBelowLine(const Common::Point &point, const Common::Rect &rect) const {
|
|
if (rect.top || rect.left || rect.bottom || rect.right) {
|
|
Common::Rational res(rect.height() * (point.x - rect.left), rect.width());
|
|
|
|
return (bool)(point.y > (rect.top + res.toInt()));
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool Scene::rectIntersect(int32 x, int32 y, int32 x1, int32 y1, int32 x2, int32 y2, int32 x3, int32 y3) const {
|
|
return (x <= x3 && x1 >= x2) && (y <= y3 && y1 >= y2);
|
|
}
|
|
|
|
void Scene::adjustCoordinates(Common::Point *point) {
|
|
if (!_ws)
|
|
error("[Scene::adjustCoordinates] WorldStats not initialized properly!");
|
|
|
|
point->x = _ws->xLeft + getCursor()->position().x;
|
|
point->y = _ws->yTop + getCursor()->position().y;
|
|
}
|
|
|
|
Actor* Scene::getActor(ActorIndex index) {
|
|
if (!_ws)
|
|
error("[Scene::getActor] WorldStats not initialized properly!");
|
|
|
|
ActorIndex computedIndex = (index != kActorInvalid) ? index : getSharedData()->getPlayerIndex();
|
|
|
|
if (computedIndex < 0 || computedIndex >= (int16)_ws->actors.size())
|
|
error("[Scene::getActor] Invalid actor index: %d ([0-%d] allowed)", computedIndex, _ws->actors.size() - 1);
|
|
|
|
return _ws->actors[computedIndex];
|
|
}
|
|
|
|
bool Scene::updateSceneCoordinates(int32 tX, int32 tY, int32 A0, bool checkSceneCoords, int32 *param) {
|
|
if (!_ws)
|
|
error("[Scene::updateSceneCoordinates] WorldStats not initialized properly!");
|
|
|
|
Common::Rect *sr = &_ws->sceneRects[_ws->sceneRectIdx];
|
|
|
|
int16 *targetX = &_ws->coordinates[0];
|
|
int16 *targetY = &_ws->coordinates[1];
|
|
int16 *coord3 = &_ws->coordinates[2];
|
|
|
|
*targetX = (int16)tX;
|
|
*targetY = (int16)tY;
|
|
|
|
*coord3 = (int16)A0;
|
|
|
|
// Adjust coordinates
|
|
if (checkSceneCoords)
|
|
if (*targetX + 640 > _ws->width)
|
|
*targetX = _ws->width - 640;
|
|
|
|
if (*targetX < sr->left)
|
|
*targetX = sr->left;
|
|
|
|
if (*targetY < sr->top)
|
|
*targetY = sr->top;
|
|
|
|
if (*targetX + 640 > sr->right)
|
|
*targetX = sr->right - 640;
|
|
|
|
if (*targetY + 480 > sr->bottom)
|
|
*targetY = sr->bottom - 480;
|
|
|
|
if (checkSceneCoords)
|
|
if (*targetY + 480 > _ws->height)
|
|
*targetY = _ws->height - 480;
|
|
|
|
// Adjust scene offsets & coordinates
|
|
getSharedData()->setSceneOffset(0);
|
|
getSharedData()->setSceneCoords(Common::Point(_ws->xLeft, _ws->yTop));
|
|
|
|
int32 diffX = *targetX - _ws->xLeft;
|
|
int32 diffY = *targetY - _ws->yTop;
|
|
|
|
if (abs(diffX) <= abs(diffY)) {
|
|
if (_ws->yTop > *targetY)
|
|
*coord3 = -*coord3;
|
|
|
|
if (diffY)
|
|
getSharedData()->setSceneOffsetAdd((int16)Common::Rational(*coord3 * diffX, diffY).toInt());
|
|
|
|
if (param != NULL && abs(diffY) <= abs(*coord3)) {
|
|
*targetX = -1;
|
|
*param = 0;
|
|
return true;
|
|
}
|
|
} else {
|
|
if (_ws->xLeft > *targetX)
|
|
*coord3 = -*coord3;
|
|
|
|
getSharedData()->setSceneOffsetAdd((int16)Common::Rational(*coord3 * diffY, diffX).toInt());
|
|
|
|
if (param != NULL && abs(diffX) <= abs(*coord3)) {
|
|
*targetX = -1;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
int32 Scene::findActionArea(ActionAreaType type, const Common::Point &pt, bool highlight) {
|
|
if (!_ws)
|
|
error("[Scene::findActionArea] WorldStats not initialized properly!");
|
|
|
|
if (!_polygons)
|
|
error("[Scene::findActionArea] Polygons not initialized properly!");
|
|
|
|
switch (type) {
|
|
default:
|
|
return type - 2;
|
|
|
|
case kActionAreaType1:
|
|
if (_ws->actions.size() < 1)
|
|
return -1;
|
|
|
|
for (int32 i = _ws->actions.size() - 1; i >= 0; i--) {
|
|
ActionArea *area = _ws->actions[i];
|
|
|
|
if (!(area->flags & 1))
|
|
continue;
|
|
|
|
if (g_debugPolygons && highlight) {
|
|
// Highlight each polygon as it gets checked for action
|
|
debugHighlightPolygon(area->polygonIndex);
|
|
}
|
|
|
|
bool found = false;
|
|
|
|
// Iterate over flagNum
|
|
for (uint32 j = 0; j < 10; j++) {
|
|
if (!area->flagNums[j])
|
|
break; // We stop as soon as a flag is 0
|
|
|
|
bool flagSet = false;
|
|
if (area->flagNums[j] <= 0)
|
|
flagSet = _vm->isGameFlagNotSet((GameFlag)-area->flagNums[j]);
|
|
else
|
|
flagSet = _vm->isGameFlagSet((GameFlag)area->flagNums[j]);
|
|
|
|
if (!flagSet) {
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!found && _polygons->get(area->polygonIndex).contains(pt))
|
|
return i;
|
|
}
|
|
break;
|
|
|
|
case kActionAreaType2:
|
|
if (_ws->actions.size() < 1)
|
|
return -1;
|
|
|
|
for (int32 i = _ws->actions.size() - 1; i >= 0; i--) {
|
|
ActionArea *area = _ws->actions[i];
|
|
|
|
bool found = false;
|
|
|
|
// Iterate over flagNum
|
|
for (uint32 j = 0; j < 10; j++) {
|
|
if (!area->flagNums[j])
|
|
continue; // We skip over null flags
|
|
|
|
bool flagSet = false;
|
|
if (area->flagNums[j] <= 0)
|
|
flagSet = _vm->isGameFlagNotSet((GameFlag)-area->flagNums[j]);
|
|
else
|
|
flagSet = _vm->isGameFlagSet((GameFlag)area->flagNums[j]);
|
|
|
|
if (!flagSet) {
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!found && _polygons->get(area->polygonIndex).contains(pt))
|
|
return i;
|
|
}
|
|
break;
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
void Scene::changePlayer(ActorIndex index) {
|
|
if (!_ws)
|
|
error("[Scene::changePlayer] WorldStats not initialized properly!");
|
|
|
|
switch (index) {
|
|
default:
|
|
if (_ws->chapter == kChapter9) {
|
|
changePlayerUpdate(index);
|
|
|
|
getActor(index)->show();
|
|
}
|
|
|
|
getSharedData()->setPlayerIndex(index);
|
|
break;
|
|
|
|
case 1:
|
|
if (_ws->chapter == kChapter9) {
|
|
changePlayerUpdate(index);
|
|
|
|
getScreen()->setPalette(_ws->graphicResourceIds[0]);
|
|
_ws->currentPaletteId = _ws->graphicResourceIds[0];
|
|
getScreen()->setGammaLevel(_ws->graphicResourceIds[0]);
|
|
_vm->setGameFlag(kGameFlag635);
|
|
_vm->clearGameFlag(kGameFlag636);
|
|
_vm->clearGameFlag(kGameFlag637);
|
|
|
|
getActor(index)->show();
|
|
}
|
|
|
|
getSharedData()->setPlayerIndex(index);
|
|
break;
|
|
|
|
case 2:
|
|
if (_ws->chapter == kChapter9) {
|
|
changePlayerUpdate(index);
|
|
|
|
getScreen()->setPalette(_ws->graphicResourceIds[1]);
|
|
_ws->currentPaletteId = _ws->graphicResourceIds[1];
|
|
getScreen()->setGammaLevel(_ws->graphicResourceIds[1]);
|
|
_vm->setGameFlag(kGameFlag636);
|
|
_vm->clearGameFlag(kGameFlag635);
|
|
_vm->clearGameFlag(kGameFlag637);
|
|
|
|
getActor(index)->show();
|
|
}
|
|
|
|
getSharedData()->setPlayerIndex(index);
|
|
break;
|
|
|
|
case 3:
|
|
if (_ws->chapter == kChapter9) {
|
|
changePlayerUpdate(index);
|
|
|
|
getScreen()->setPalette(_ws->graphicResourceIds[2]);
|
|
_ws->currentPaletteId = _ws->graphicResourceIds[2];
|
|
getScreen()->setGammaLevel(_ws->graphicResourceIds[2]);
|
|
_vm->setGameFlag(kGameFlag637);
|
|
_vm->clearGameFlag(kGameFlag635);
|
|
_vm->clearGameFlag(kGameFlag636);
|
|
|
|
getActor(index)->show();
|
|
}
|
|
|
|
getActor(index)->show();
|
|
getSharedData()->setPlayerIndex(index);
|
|
break;
|
|
|
|
case 666:
|
|
getScreen()->setupTransTables(3, _ws->graphicResourceIds[50], _ws->graphicResourceIds[49], _ws->graphicResourceIds[48]);
|
|
|
|
// Save scene data
|
|
getSharedData()->saveCursorResources((ResourceId *)&_ws->cursorResources, sizeof(_ws->cursorResources));
|
|
getSharedData()->saveSceneFonts(_ws->font1, _ws->font2, _ws->font3);
|
|
getSharedData()->saveSmallCursor(_ws->smallCurDown, _ws->smallCurUp);
|
|
getSharedData()->saveEncounterFrameBackground(_ws->encounterFrameBg);
|
|
|
|
// Setup new values
|
|
for (uint32 i = 0; i < 11; i++)
|
|
_ws->cursorResources[i] = _ws->graphicResourceIds[20 + i];
|
|
|
|
_ws->font1 = _ws->graphicResourceIds[35];
|
|
_ws->font2 = _ws->graphicResourceIds[37];
|
|
_ws->font3 = _ws->graphicResourceIds[36];
|
|
|
|
_ws->smallCurUp = _ws->graphicResourceIds[33];
|
|
_ws->smallCurDown = _ws->graphicResourceIds[34];
|
|
_ws->encounterFrameBg = _ws->graphicResourceIds[32];
|
|
break;
|
|
|
|
case 667:
|
|
getScreen()->setupTransTables(3, _ws->cellShadeMask1, _ws->cellShadeMask2, _ws->cellShadeMask3);
|
|
|
|
// Load scene data
|
|
getSharedData()->loadCursorResources((ResourceId *)&_ws->cursorResources, sizeof(_ws->cursorResources));
|
|
getSharedData()->loadSceneFonts(&_ws->font1, &_ws->font2, &_ws->font3);
|
|
getSharedData()->loadSmallCursor(&_ws->smallCurDown, &_ws->smallCurUp);
|
|
getSharedData()->loadEncounterFrameBackground(&_ws->encounterFrameBg);
|
|
|
|
// Reset cursor
|
|
getCursor()->set(_ws->cursorResources[kCursorResourceMagnifyingGlass], 0, kCursorAnimationNone);
|
|
break;
|
|
|
|
case 668:
|
|
getActor(11)->setPosition(2300, 100, kDirectionN, 0);
|
|
getSharedData()->setChapter2Counter(6, 0);
|
|
getSharedData()->setFlag(kFlag1, false);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void Scene::changePlayerUpdate(ActorIndex index) {
|
|
Actor *actor = getActor(index);
|
|
Actor *player = getActor();
|
|
|
|
actor->setPosition(player->getPoint1()->x + player->getPoint2()->x, player->getPoint1()->y + player->getPoint2()->y, player->getDirection(), 0);
|
|
player->hide();
|
|
|
|
actor->inventory.copyFrom(player->inventory);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
// Scene drawing
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void Scene::preload() {
|
|
if (!Config.showSceneLoading || _vm->checkGameVersion("Demo"))
|
|
return;
|
|
|
|
SceneTitle *title = new SceneTitle(_vm);
|
|
title->load();
|
|
getCursor()->hide();
|
|
|
|
do {
|
|
title->update(_vm->getTick());
|
|
|
|
getScreen()->copyBackBufferToScreen();
|
|
g_system->updateScreen();
|
|
|
|
g_system->delayMillis(10);
|
|
|
|
// Poll events (this ensure we don't freeze the screen)
|
|
Common::Event ev;
|
|
_vm->getEventManager()->pollEvent(ev);
|
|
|
|
} while (!title->loadingComplete());
|
|
|
|
delete title;
|
|
}
|
|
|
|
bool Scene::drawScene() {
|
|
if (!_ws)
|
|
error("[Scene::drawScene] WorldStats not initialized properly!");
|
|
|
|
_vm->screen()->clearGraphicsInQueue();
|
|
|
|
if (getSharedData()->getFlag(kFlagSkipDrawScene)) {
|
|
_vm->screen()->fillRect(0, 0, 640, 480, 0);
|
|
getCursor()->hide();
|
|
|
|
return false;
|
|
}
|
|
|
|
// Draw scene background
|
|
getScreen()->draw(_ws->backgroundImage, 0, Common::Point(-_ws->xLeft, -_ws->yTop), kDrawFlagNone, false);
|
|
|
|
// Draw actors on the update list
|
|
buildUpdateList();
|
|
processUpdateList();
|
|
|
|
if (_ws->chapter == kChapter11)
|
|
checkVisibleActorsPriority();
|
|
|
|
// Queue updates
|
|
for (uint32 i = 0; i < _ws->actors.size(); i++)
|
|
_ws->actors[i]->draw();
|
|
|
|
for (uint32 i = 0; i < _ws->objects.size(); i++)
|
|
_ws->objects[i]->draw();
|
|
|
|
Actor *player = getActor();
|
|
if (player->getStatus() == kActorStatusShowingInventory || player->getStatus() == kActorStatus10)
|
|
player->drawInventory();
|
|
else
|
|
player->setNumberFlag01(0);
|
|
|
|
_vm->screen()->drawGraphicsInQueue();
|
|
|
|
// Show debug information
|
|
if (g_debugScrolling)
|
|
debugScreenScrolling();
|
|
if (g_debugActors)
|
|
debugShowActors();
|
|
if (g_debugPolygons)
|
|
debugShowPolygons();
|
|
if (g_debugObjects)
|
|
debugShowObjects();
|
|
if (g_debugSceneRects)
|
|
debugShowSceneRects();
|
|
|
|
return false;
|
|
}
|
|
|
|
bool Scene::updateListCompare(const UpdateItem &item1, const UpdateItem &item2) {
|
|
return (item1.priority < item2.priority);
|
|
}
|
|
|
|
void Scene::buildUpdateList() {
|
|
if (!_ws)
|
|
error("[Scene::buildUpdateList] WorldStats not initialized properly!");
|
|
|
|
_updateList.clear();
|
|
|
|
for (uint32 i = 0; i < _ws->actors.size(); i++) {
|
|
Actor *actor = _ws->actors[i];
|
|
|
|
if (actor->isVisible()) {
|
|
UpdateItem item;
|
|
item.index = i;
|
|
item.priority = actor->getPoint1()->y + actor->getPoint2()->y;
|
|
|
|
_updateList.push_back(item);
|
|
}
|
|
}
|
|
|
|
// Sort the list (the original uses qsort, so we may have to revert to that if our sort isn't behaving the same)
|
|
Common::sort(_updateList.begin(), _updateList.end(), &Scene::updateListCompare);
|
|
}
|
|
|
|
void Scene::processUpdateList() {
|
|
if (!_ws)
|
|
error("[Scene::processUpdateList] WorldStats not initialized properly!");
|
|
|
|
for (uint32 i = 0; i < _updateList.size(); i++) {
|
|
Actor *actor = getActor(_updateList[i].index);
|
|
int32 priority = _updateList[i].priority;
|
|
|
|
// Check priority
|
|
if (priority < 0) {
|
|
actor->setPriority(abs(priority));
|
|
continue;
|
|
}
|
|
|
|
actor->setPriority(3);
|
|
|
|
if (actor->getField944() == 1 || actor->getField944() == 4) {
|
|
actor->setPriority(1);
|
|
} else {
|
|
actor->setField938(1);
|
|
actor->setField934(0);
|
|
Common::Point sum = *actor->getPoint1() + *actor->getPoint2();
|
|
|
|
int16 bottomRight = actor->getPoint1()->y + actor->getBoundingRect()->bottom + 4;
|
|
|
|
if (_ws->chapter == kChapter11 && _updateList[i].index != getSharedData()->getPlayerIndex())
|
|
bottomRight += 20;
|
|
|
|
// Our actor rect
|
|
Common::Rect actorRect(actor->getPoint1()->x, actor->getPoint1()->y, actor->getPoint1()->x + actor->getBoundingRect()->right, bottomRight);
|
|
|
|
// Process objects
|
|
for (uint32 j = 0; j < _ws->objects.size(); j++) {
|
|
Object *object = _ws->objects[j];
|
|
|
|
// Skip hidden objects
|
|
if (!object->isOnScreen())
|
|
continue;
|
|
|
|
// Check that the rects are contained
|
|
if (!rectIntersect(object->x,
|
|
object->y,
|
|
object->x + object->getBoundingRect()->right,
|
|
object->y + object->getBoundingRect()->bottom,
|
|
actor->getPoint1()->x,
|
|
actor->getPoint1()->y,
|
|
actor->getPoint1()->x + actor->getBoundingRect()->right,
|
|
bottomRight)) {
|
|
|
|
if ((BYTE1(object->flags) & kObjectFlag20) && !(BYTE1(object->flags) & kObjectFlag80))
|
|
BYTE1(object->flags) = BYTE1(object->flags) | kObjectFlag40;
|
|
|
|
continue;
|
|
}
|
|
|
|
// Check if it intersects with either the object rect or the related polygon
|
|
bool isMasked = false;
|
|
if (object->flags & kObjectFlag2) {
|
|
isMasked = !pointBelowLine(sum, *object->getRect());
|
|
} else if (object->flags & kObjectFlag40) {
|
|
Polygon poly = _polygons->get(object->getPolygonIndex());
|
|
isMasked = poly.contains(sum);
|
|
}
|
|
|
|
// Adjust object flags
|
|
if ((BYTE1(object->flags) & kObjectFlag80) || isMasked) {
|
|
if (BYTE1(object->flags) & kObjectFlag20)
|
|
BYTE1(object->flags) = (BYTE1(object->flags) & kObjectFlagBF) | kObjectFlag80;
|
|
} else {
|
|
if (BYTE1(object->flags) & kObjectFlag20) {
|
|
BYTE1(object->flags) = BYTE1(object->flags) | kObjectFlag40;
|
|
}
|
|
}
|
|
|
|
if (LOBYTE(object->flags) & kObjectFlag4) {
|
|
if (isMasked) {
|
|
if (actor->flags & kActorFlagMasked)
|
|
error("[Scene::processUpdateList] Assigning mask to masked character (%s)", actor->getName());
|
|
|
|
// We are masked by the object!
|
|
actor->setObjectIndex(j);
|
|
actor->flags |= kActorFlagMasked;
|
|
}
|
|
} else {
|
|
if (isMasked) {
|
|
if (actor->getPriority() < object->getPriority()) {
|
|
actor->setField934(1);
|
|
actor->setPriority(object->getPriority() + 3);
|
|
|
|
if (i > 0) {
|
|
// Update actor priority up to current index
|
|
for (uint32 k = 0; k < i; k++) {
|
|
Actor *previousActor = getWorld()->actors[_updateList[k].index];
|
|
|
|
if (getWorld()->chapter != kChapter2 || previousActor->getField944() != 1) {
|
|
if (previousActor->getPriority() == actor->getPriority())
|
|
actor->setPriority(actor->getPriority() - 1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
if (actor->getPriority() > object->getPriority() || actor->getPriority() == 1) {
|
|
actor->setField934(1);
|
|
actor->setPriority(object->getPriority() - 1);
|
|
|
|
if (i > 0) {
|
|
|
|
// Update actor priority up to current index
|
|
for (uint32 k = 0; k < i; k++) {
|
|
Actor *previousActor = getWorld()->actors[_updateList[k].index];
|
|
|
|
if (previousActor->getField944() != 1 && previousActor->getField944() != 4) {
|
|
Actor *actorCheck = getWorld()->actors[k];
|
|
|
|
if (rectIntersect(actorCheck->getPoint1()->x,
|
|
actorCheck->getPoint1()->y,
|
|
actorCheck->getPoint1()->x + actorCheck->getBoundingRect()->bottom,
|
|
actorCheck->getPoint1()->y + actorCheck->getBoundingRect()->right,
|
|
actor->getPoint1()->x,
|
|
actor->getPoint1()->y,
|
|
actor->getPoint1()->x + actor->getBoundingRect()->right,
|
|
bottomRight)) {
|
|
if (previousActor->getPriority() == actor->getPriority())
|
|
actor->setPriority(actor->getPriority() - 1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} // end processing objects
|
|
|
|
// Update all other actors
|
|
for (uint32 k = 0; k < _updateList.size(); k++) {
|
|
Actor *actor2 = getActor(_updateList[k].index);
|
|
|
|
if (actor2->isVisible() && actor2->getField944() != 1 && actor2->getField944() != 4 && _updateList[k].index != _updateList[i].index) {
|
|
|
|
Common::Rect actor2Rect(actor2->getPoint1()->x, actor2->getPoint1()->y, actor2->getPoint1()->x + actor2->getBoundingRect()->right, actor2->getPoint1()->y + actor2->getBoundingRect()->bottom);
|
|
|
|
if (actor2Rect.contains(actorRect)) {
|
|
|
|
// Inferior
|
|
if ((actor2->getPoint1()->y + actor2->getPoint2()->y) > (actor->getPoint1()->y + actor->getPoint2()->y)) {
|
|
if (actor->getPriority() <= actor2->getPriority()) {
|
|
if (actor->getField934() || actor2->getNumberValue01()) {
|
|
if (!actor2->getNumberValue01())
|
|
actor2->setPriority(actor->getPriority() - 1);
|
|
} else {
|
|
actor->setPriority(actor2->getPriority() + 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Superior
|
|
if ((actor2->getPoint1()->y + actor2->getPoint2()->y) < (actor->getPoint1()->y + actor->getPoint2()->y)) {
|
|
if (actor->getPriority() >= actor2->getPriority()) {
|
|
if (actor->getField934() || actor2->getNumberValue01()) {
|
|
if (!actor2->getNumberValue01())
|
|
actor2->setPriority(actor->getPriority() + 1);
|
|
} else {
|
|
actor->setPriority(actor2->getPriority() - 1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (actor->shouldInvertPriority())
|
|
getActor(actor->getNextActorIndex())->setPriority(-actor->getPriority());
|
|
}
|
|
} // end processing actors
|
|
|
|
|
|
// Go through the list from the end
|
|
if (_updateList.size() > 1) {
|
|
for (int i = _updateList.size() - 1; i >= 0; --i) {
|
|
// Get the actor
|
|
Actor *actor1 = getActor(_updateList[i].index);
|
|
|
|
// Skip hidden actors
|
|
if (!actor1->isVisible())
|
|
continue;
|
|
|
|
if (actor1->getField944() != 1 && actor1->getField944() != 4) {
|
|
// Process all previous actors
|
|
if (i > 0) {
|
|
for (int j = i - 1; j >= 0; --j) {
|
|
Actor *actor2 = getActor(_updateList[j].index);
|
|
|
|
if (actor2->getField944() != 1 && actor2->getField944() != 4) {
|
|
|
|
// Process priorities
|
|
if (actor2->getPriority() <= actor1->getPriority()) {
|
|
if (rectIntersect(actor1->getPoint1()->x,
|
|
actor1->getPoint1()->y,
|
|
actor1->getPoint1()->x + actor1->getBoundingRect()->right,
|
|
actor1->getPoint1()->y + actor1->getBoundingRect()->bottom,
|
|
actor2->getPoint1()->x,
|
|
actor2->getPoint1()->y,
|
|
actor2->getPoint1()->x + actor2->getBoundingRect()->right,
|
|
actor2->getPoint1()->y + actor2->getBoundingRect()->bottom)) {
|
|
actor2->setPriority(actor1->getPriority() + 1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void Scene::checkVisibleActorsPriority() {
|
|
for (uint i = 2; i < 9; i++)
|
|
if (getActor(i)->isVisible())
|
|
adjustActorPriority(i);
|
|
|
|
for (uint i = 16; i < 18; i++)
|
|
if (getActor(i)->isVisible())
|
|
adjustActorPriority(i);
|
|
}
|
|
|
|
void Scene::adjustActorPriority(ActorIndex index) {
|
|
Actor *actor0 = getActor(0);
|
|
Actor *actor = getActor(index);
|
|
|
|
if (rectIntersect(actor0->getPoint1()->x,
|
|
actor0->getPoint1()->y,
|
|
actor0->getPoint1()->x + actor0->getBoundingRect()->right,
|
|
actor0->getPoint1()->y + actor0->getBoundingRect()->bottom + 4,
|
|
actor->getPoint1()->x,
|
|
actor->getPoint1()->y,
|
|
actor->getPoint1()->x + actor0->getBoundingRect()->right,
|
|
actor->getPoint1()->y + actor0->getBoundingRect()->bottom)) {
|
|
if (actor->getPriority() < actor0->getPriority())
|
|
actor0->setPriority(actor->getPriority());
|
|
}
|
|
}
|
|
|
|
void Scene::drawRain() {
|
|
if (!_ws)
|
|
error("[Scene::drawRain] WorldStats not initialized properly!");
|
|
|
|
if (getSharedData()->getFlag(kFlagSkipDrawScene))
|
|
return;
|
|
|
|
for (int16 y = 0; y < 512; y = y + 64) {
|
|
for (int16 x = 0; x < 704; x = x + 64) {
|
|
getScreen()->draw(MAKE_RESOURCE(kResourcePackShared, 58), (uint32)_chapter5RainFrameIndex, Common::Point(x + (_ws->xLeft % 64) / 8, y + (_ws->yTop % 64) / 8));
|
|
}
|
|
}
|
|
|
|
_chapter5RainFrameIndex = (_chapter5RainFrameIndex + 1) % (int32)GraphicResource::getFrameCount(_vm, MAKE_RESOURCE(kResourcePackShared, 58));
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
// Debug
|
|
//////////////////////////////////////////////////////////////////////////
|
|
void Scene::debugScreenScrolling() {
|
|
if (!_ws)
|
|
error("[Scene::debugScreenScrolling] WorldStats not initialized properly!");
|
|
|
|
Common::Rect rect = GraphicResource::getFrameRect(_vm, _ws->backgroundImage, 0);
|
|
|
|
// Horizontal scrolling
|
|
if (getCursor()->position().x < SCREEN_EDGES && _ws->xLeft >= SCROLL_STEP)
|
|
_ws->xLeft -= SCROLL_STEP;
|
|
else if (getCursor()->position().x > (640 - SCREEN_EDGES) && _ws->xLeft <= (rect.width() - (640 + SCROLL_STEP)))
|
|
_ws->xLeft += SCROLL_STEP;
|
|
|
|
// Vertical scrolling
|
|
if (getCursor()->position().y < SCREEN_EDGES && _ws->yTop >= SCROLL_STEP)
|
|
_ws->yTop -= SCROLL_STEP;
|
|
else if (getCursor()->position().y > (480 - SCREEN_EDGES) && _ws->yTop <= (rect.height() - (480 + SCROLL_STEP)))
|
|
_ws->yTop += SCROLL_STEP;
|
|
}
|
|
|
|
// WALK REGION DEBUG
|
|
void Scene::debugShowWalkRegion(Polygon *poly) {
|
|
Graphics::Surface surface;
|
|
surface.create((uint16)poly->boundingRect.width() + 1,
|
|
(uint16)poly->boundingRect.height() + 1,
|
|
Graphics::PixelFormat::createFormatCLUT8());
|
|
|
|
// Draw all lines in Polygon
|
|
for (uint32 i = 0; i < poly->count(); i++) {
|
|
surface.drawLine(
|
|
poly->points[i].x - poly->boundingRect.left,
|
|
poly->points[i].y - poly->boundingRect.top,
|
|
poly->points[(i+1) % poly->count()].x - poly->boundingRect.left,
|
|
poly->points[(i+1) % poly->count()].y - poly->boundingRect.top, 0x3A);
|
|
}
|
|
|
|
getScreen()->copyToBackBufferClipped(&surface, poly->boundingRect.left, poly->boundingRect.top);
|
|
|
|
surface.free();
|
|
}
|
|
|
|
// POLYGONS DEBUG
|
|
void Scene::debugShowPolygons() {
|
|
if (!_polygons)
|
|
error("[Scene::debugShowPolygons] Polygons not initialized properly!");
|
|
|
|
for (uint32 p = 0; p < _polygons->size(); p++)
|
|
debugShowPolygon(p);
|
|
}
|
|
|
|
void Scene::debugShowPolygon(uint32 index, uint32 color) {
|
|
if (!_polygons)
|
|
error("[Scene::debugShowPolygon] Polygons not initialized properly");
|
|
|
|
if (index >= _polygons->size() - 1)
|
|
return;
|
|
|
|
Graphics::Surface surface;
|
|
Polygon poly = _polygons->get(index);
|
|
surface.create((uint16)poly.boundingRect.width() + 1,
|
|
(uint16)poly.boundingRect.height() + 1,
|
|
Graphics::PixelFormat::createFormatCLUT8());
|
|
|
|
// Draw all lines in Polygon
|
|
for (uint32 i = 0; i < poly.count(); i++) {
|
|
surface.drawLine(poly.points[i].x - poly.boundingRect.left,
|
|
poly.points[i].y - poly.boundingRect.top,
|
|
poly.points[(i+1) % poly.count()].x - poly.boundingRect.left,
|
|
poly.points[(i+1) % poly.count()].y - poly.boundingRect.top,
|
|
color);
|
|
}
|
|
|
|
getScreen()->copyToBackBufferClipped(&surface, poly.boundingRect.left, poly.boundingRect.top);
|
|
|
|
surface.free();
|
|
}
|
|
|
|
void Scene::debugHighlightPolygon(uint32 index) {
|
|
debugShowPolygon(index, 0x12);
|
|
getScreen()->copyBackBufferToScreen();
|
|
g_system->updateScreen();
|
|
}
|
|
|
|
// SCENE RECTS DEBUG
|
|
void Scene::debugShowSceneRects() {
|
|
if (!_ws)
|
|
error("[Scene::debugShowObjects] WorldStats not initialized properly!");
|
|
|
|
for (uint32 i = 0; i < ARRAYSIZE(_ws->sceneRects); i++)
|
|
getScreen()->drawRect(_ws->sceneRects[i]);
|
|
}
|
|
|
|
// OBJECT DEBUGGING
|
|
void Scene::debugShowObjects() {
|
|
if (!_ws)
|
|
error("[Scene::debugShowObjects] WorldStats not initialized properly!");
|
|
|
|
for (uint32 p = 0; p < _ws->objects.size(); p++) {
|
|
Graphics::Surface surface;
|
|
Object *object = _ws->objects[p];
|
|
|
|
if (object->isOnScreen()) {
|
|
surface.create((uint16)object->getBoundingRect()->width() + 1,
|
|
(uint16)object->getBoundingRect()->height() + 1,
|
|
Graphics::PixelFormat::createFormatCLUT8());
|
|
surface.frameRect(*object->getBoundingRect(), 0x22);
|
|
getScreen()->copyToBackBufferClipped(&surface, object->x, object->y);
|
|
}
|
|
|
|
surface.free();
|
|
}
|
|
}
|
|
|
|
// ACTOR DEBUGGING
|
|
void Scene::debugShowActors() {
|
|
if (!_ws)
|
|
error("[Scene::debugShowActors] WorldStats not initialized properly!");
|
|
|
|
for (uint32 p = 0; p < _ws->actors.size(); p++) {
|
|
Graphics::Surface surface;
|
|
Actor *a = _ws->actors[p];
|
|
|
|
if (a->isOnScreen()) {
|
|
surface.create((uint16)a->getBoundingRect()->width() + 1,
|
|
(uint16)a->getBoundingRect()->height() + 1,
|
|
Graphics::PixelFormat::createFormatCLUT8());
|
|
surface.frameRect(*a->getBoundingRect(), 0x128);
|
|
getScreen()->copyToBackBufferClipped(&surface, a->getPoint1()->x, a->getPoint1()->y);
|
|
}
|
|
|
|
surface.free();
|
|
}
|
|
}
|
|
|
|
} // end of namespace Asylum
|