2011-06-30 09:55:59 +00:00
|
|
|
/* ScummVM - Graphic Adventure Engine
|
|
|
|
*
|
|
|
|
* ScummVM is the legal property of its developers, whose names
|
|
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
|
|
* file distributed with this source distribution.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
2014-02-18 01:34:22 +00:00
|
|
|
*
|
2011-06-30 09:55:59 +00:00
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
2014-02-18 01:34:22 +00:00
|
|
|
*
|
2011-06-30 09:55:59 +00:00
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2013-06-09 14:13:23 +00:00
|
|
|
#include "neverhood/console.h"
|
2011-06-30 09:55:59 +00:00
|
|
|
#include "neverhood/scene.h"
|
2016-05-21 10:44:27 +00:00
|
|
|
#include "neverhood/smackerplayer.h"
|
2011-06-30 09:55:59 +00:00
|
|
|
|
|
|
|
namespace Neverhood {
|
|
|
|
|
2013-01-10 23:17:16 +00:00
|
|
|
Scene::Scene(NeverhoodEngine *vm, Module *parentModule)
|
2013-05-05 10:09:13 +00:00
|
|
|
: Entity(vm, 0), _parentModule(parentModule), _dataResource(vm), _hitRects(NULL),
|
|
|
|
_mouseCursorWasVisible(true) {
|
2013-07-14 17:01:47 +00:00
|
|
|
|
2013-01-21 11:34:50 +00:00
|
|
|
_isKlaymenBusy = false;
|
2012-11-14 12:21:13 +00:00
|
|
|
_doConvertMessages = false;
|
2011-06-30 09:55:59 +00:00
|
|
|
_messageList = NULL;
|
2011-07-08 19:09:33 +00:00
|
|
|
_rectType = 0;
|
2011-06-30 09:55:59 +00:00
|
|
|
_mouseClickPos.x = 0;
|
|
|
|
_mouseClickPos.y = 0;
|
|
|
|
_mouseClicked = false;
|
2011-07-08 19:09:33 +00:00
|
|
|
_rectList = NULL;
|
2013-01-21 11:34:50 +00:00
|
|
|
_klaymen = NULL;
|
2011-07-11 12:21:56 +00:00
|
|
|
_mouseCursor = NULL;
|
2011-06-30 09:55:59 +00:00
|
|
|
_palette = NULL;
|
2011-07-03 19:26:27 +00:00
|
|
|
_background = NULL;
|
2013-01-10 23:17:16 +00:00
|
|
|
clearHitRects();
|
|
|
|
clearCollisionSprites();
|
2011-07-03 12:09:48 +00:00
|
|
|
_vm->_screen->setFps(24);
|
2012-10-09 18:30:39 +00:00
|
|
|
_vm->_screen->setSmackerDecoder(NULL);
|
|
|
|
_canAcceptInput = true;
|
2011-06-30 09:55:59 +00:00
|
|
|
_messageList2 = NULL;
|
2011-07-03 19:26:27 +00:00
|
|
|
_smackerPlayer = NULL;
|
2012-10-09 18:30:39 +00:00
|
|
|
_isMessageListBusy = false;
|
2011-06-30 09:55:59 +00:00
|
|
|
_messageValue = -1;
|
2013-09-26 20:54:23 +00:00
|
|
|
_messageListStatus = 0;
|
|
|
|
_messageListCount = 0;
|
|
|
|
_messageListIndex = 0;
|
2013-07-14 17:01:47 +00:00
|
|
|
|
2013-09-14 09:39:12 +00:00
|
|
|
_backgroundFileHash = _cursorFileHash = 0;
|
|
|
|
|
2011-06-30 09:55:59 +00:00
|
|
|
SetUpdateHandler(&Scene::update);
|
|
|
|
SetMessageHandler(&Scene::handleMessage);
|
2013-07-14 17:01:47 +00:00
|
|
|
|
2012-10-19 21:17:10 +00:00
|
|
|
_vm->_screen->clearRenderQueue();
|
2011-06-30 09:55:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Scene::~Scene() {
|
|
|
|
|
2013-06-17 07:21:16 +00:00
|
|
|
_vm->_screen->setSmackerDecoder(NULL);
|
|
|
|
|
2011-06-30 09:55:59 +00:00
|
|
|
if (_palette) {
|
|
|
|
removeEntity(_palette);
|
|
|
|
delete _palette;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delete all entities
|
|
|
|
for (Common::Array<Entity*>::iterator iter = _entities.begin(); iter != _entities.end(); iter++)
|
|
|
|
delete *iter;
|
|
|
|
|
|
|
|
// Don't delete surfaces since they always belong to an entity
|
2013-07-14 17:01:47 +00:00
|
|
|
|
2013-01-10 00:08:41 +00:00
|
|
|
// Purge the resources after each scene
|
|
|
|
_vm->_res->purgeResources();
|
2011-06-30 09:55:59 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void Scene::draw() {
|
|
|
|
if (_smackerPlayer) {
|
2011-07-04 17:58:38 +00:00
|
|
|
if (_smackerPlayer->getSurface())
|
|
|
|
_smackerPlayer->getSurface()->draw();
|
2011-06-30 09:55:59 +00:00
|
|
|
} else {
|
2012-10-18 11:23:34 +00:00
|
|
|
for (Common::Array<BaseSurface*>::iterator iter = _surfaces.begin(); iter != _surfaces.end(); iter++)
|
2011-06-30 09:55:59 +00:00
|
|
|
(*iter)->draw();
|
2013-07-14 17:01:47 +00:00
|
|
|
}
|
2011-06-30 09:55:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Scene::addEntity(Entity *entity) {
|
|
|
|
int index = 0, insertIndex = -1;
|
|
|
|
for (Common::Array<Entity*>::iterator iter = _entities.begin(); iter != _entities.end(); iter++) {
|
|
|
|
if ((*iter)->getPriority() > entity->getPriority()) {
|
|
|
|
insertIndex = index;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
index++;
|
|
|
|
}
|
|
|
|
if (insertIndex >= 0)
|
|
|
|
_entities.insert_at(insertIndex, entity);
|
|
|
|
else
|
2013-07-14 17:01:47 +00:00
|
|
|
_entities.push_back(entity);
|
2011-06-30 09:55:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool Scene::removeEntity(Entity *entity) {
|
2012-10-27 22:18:54 +00:00
|
|
|
for (uint index = 0; index < _entities.size(); index++)
|
2011-06-30 09:55:59 +00:00
|
|
|
if (_entities[index] == entity) {
|
|
|
|
_entities.remove_at(index);
|
|
|
|
return true;
|
|
|
|
}
|
2013-07-14 17:01:47 +00:00
|
|
|
return false;
|
2011-06-30 09:55:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Scene::addSurface(BaseSurface *surface) {
|
2012-10-27 22:18:54 +00:00
|
|
|
if (surface) {
|
|
|
|
int index = 0, insertIndex = -1;
|
|
|
|
for (Common::Array<BaseSurface*>::iterator iter = _surfaces.begin(); iter != _surfaces.end(); iter++) {
|
|
|
|
if ((*iter)->getPriority() > surface->getPriority()) {
|
|
|
|
insertIndex = index;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
index++;
|
2011-06-30 09:55:59 +00:00
|
|
|
}
|
2012-10-27 22:18:54 +00:00
|
|
|
if (insertIndex >= 0)
|
|
|
|
_surfaces.insert_at(insertIndex, surface);
|
|
|
|
else
|
2013-07-14 17:01:47 +00:00
|
|
|
_surfaces.push_back(surface);
|
2011-06-30 09:55:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Scene::removeSurface(BaseSurface *surface) {
|
|
|
|
for (uint index = 0; index < _surfaces.size(); index++) {
|
|
|
|
if (_surfaces[index] == surface) {
|
|
|
|
_surfaces.remove_at(index);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
2013-07-14 17:01:47 +00:00
|
|
|
return false;
|
2011-06-30 09:55:59 +00:00
|
|
|
}
|
|
|
|
|
2013-06-09 14:13:23 +00:00
|
|
|
void Scene::printSurfaces(Console *con) {
|
|
|
|
for (uint index = 0; index < _surfaces.size(); index++) {
|
|
|
|
NDrawRect drawRect = _surfaces[index]->getDrawRect();
|
|
|
|
NRect clipRect = _surfaces[index]->getClipRect();
|
|
|
|
int priority = _surfaces[index]->getPriority();
|
2014-05-27 00:04:07 +00:00
|
|
|
con->debugPrintf("%d ('%s'): Priority %d, draw rect (%d, %d, %d, %d), clip rect (%d, %d, %d, %d)\n",
|
2013-06-09 14:13:23 +00:00
|
|
|
index, _surfaces[index]->getName().c_str(), priority,
|
|
|
|
drawRect.x, drawRect.y, drawRect.x2(), drawRect.y2(),
|
|
|
|
clipRect.x1, clipRect.y1, clipRect.x2, clipRect.y2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-06-30 09:55:59 +00:00
|
|
|
Sprite *Scene::addSprite(Sprite *sprite) {
|
|
|
|
addEntity(sprite);
|
|
|
|
addSurface(sprite->getSurface());
|
|
|
|
return sprite;
|
|
|
|
}
|
|
|
|
|
2012-09-11 13:56:48 +00:00
|
|
|
void Scene::removeSprite(Sprite *sprite) {
|
|
|
|
removeSurface(sprite->getSurface());
|
|
|
|
removeEntity(sprite);
|
|
|
|
}
|
|
|
|
|
2011-06-30 09:55:59 +00:00
|
|
|
void Scene::setSurfacePriority(BaseSurface *surface, int priority) {
|
|
|
|
surface->setPriority(priority);
|
|
|
|
if (removeSurface(surface))
|
|
|
|
addSurface(surface);
|
|
|
|
}
|
|
|
|
|
2012-10-04 09:39:26 +00:00
|
|
|
void Scene::setSpriteSurfacePriority(Sprite *sprite, int priority) {
|
|
|
|
if (sprite)
|
|
|
|
setSurfacePriority(sprite->getSurface(), priority);
|
|
|
|
}
|
|
|
|
|
2011-06-30 09:55:59 +00:00
|
|
|
void Scene::deleteSprite(Sprite **sprite) {
|
2013-01-10 23:17:16 +00:00
|
|
|
removeCollisionSprite(*sprite);
|
2011-06-30 09:55:59 +00:00
|
|
|
removeSurface((*sprite)->getSurface());
|
|
|
|
removeEntity(*sprite);
|
|
|
|
delete *sprite;
|
|
|
|
*sprite = NULL;
|
|
|
|
}
|
|
|
|
|
2011-07-03 19:26:27 +00:00
|
|
|
Background *Scene::addBackground(Background *background) {
|
|
|
|
addEntity(background);
|
|
|
|
addSurface(background->getSurface());
|
|
|
|
return background;
|
|
|
|
}
|
|
|
|
|
2012-10-18 11:19:14 +00:00
|
|
|
void Scene::setBackground(uint32 fileHash) {
|
|
|
|
_background = addBackground(new Background(_vm, fileHash, 0, 0));
|
2013-09-14 09:39:12 +00:00
|
|
|
_backgroundFileHash = fileHash;
|
2011-09-16 12:23:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Scene::changeBackground(uint32 fileHash) {
|
|
|
|
_background->load(fileHash);
|
|
|
|
}
|
|
|
|
|
2011-09-19 14:46:53 +00:00
|
|
|
void Scene::setPalette(uint32 fileHash) {
|
|
|
|
_palette = fileHash ? new Palette(_vm, fileHash) : new Palette(_vm);
|
|
|
|
_palette->usePalette();
|
|
|
|
}
|
|
|
|
|
2011-09-20 09:39:24 +00:00
|
|
|
void Scene::setHitRects(uint32 id) {
|
2013-01-10 23:17:16 +00:00
|
|
|
setHitRects(_vm->_staticData->getHitRectList(id));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Scene::setHitRects(HitRectList *hitRects) {
|
|
|
|
_hitRects = hitRects;
|
2011-09-20 09:39:24 +00:00
|
|
|
}
|
|
|
|
|
2011-09-16 12:23:26 +00:00
|
|
|
Sprite *Scene::insertStaticSprite(uint32 fileHash, int surfacePriority) {
|
|
|
|
return addSprite(new StaticSprite(_vm, fileHash, surfacePriority));
|
|
|
|
}
|
|
|
|
|
2013-01-10 22:52:46 +00:00
|
|
|
void Scene::insertScreenMouse(uint32 fileHash, const NRect *mouseRect) {
|
2013-08-05 09:47:11 +00:00
|
|
|
NRect rect = NRect::make(-1, -1, -1, -1);
|
2011-09-19 09:12:43 +00:00
|
|
|
if (mouseRect)
|
|
|
|
rect = *mouseRect;
|
2013-01-17 08:27:38 +00:00
|
|
|
insertMouse(new Mouse(_vm, fileHash, rect));
|
2013-09-14 09:39:12 +00:00
|
|
|
_cursorFileHash = fileHash;
|
2011-09-19 09:12:43 +00:00
|
|
|
}
|
|
|
|
|
2013-01-10 22:52:46 +00:00
|
|
|
void Scene::insertPuzzleMouse(uint32 fileHash, int16 x1, int16 x2) {
|
2013-01-17 08:27:38 +00:00
|
|
|
insertMouse(new Mouse(_vm, fileHash, x1, x2));
|
2013-09-14 09:39:12 +00:00
|
|
|
_cursorFileHash = fileHash;
|
2011-09-19 09:12:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Scene::insertNavigationMouse(uint32 fileHash, int type) {
|
2013-01-17 08:27:38 +00:00
|
|
|
insertMouse(new Mouse(_vm, fileHash, type));
|
2013-09-14 09:39:12 +00:00
|
|
|
_cursorFileHash = fileHash;
|
2011-09-19 09:12:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Scene::showMouse(bool visible) {
|
|
|
|
_mouseCursor->getSurface()->setVisible(visible);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Scene::changeMouseCursor(uint32 fileHash) {
|
|
|
|
_mouseCursor->load(fileHash);
|
|
|
|
_mouseCursor->updateCursor();
|
|
|
|
}
|
|
|
|
|
2011-07-29 21:12:59 +00:00
|
|
|
SmackerPlayer *Scene::addSmackerPlayer(SmackerPlayer *smackerPlayer) {
|
|
|
|
addEntity(smackerPlayer);
|
|
|
|
addSurface(smackerPlayer->getSurface());
|
|
|
|
return smackerPlayer;
|
|
|
|
}
|
|
|
|
|
2011-06-30 09:55:59 +00:00
|
|
|
void Scene::update() {
|
|
|
|
|
2012-10-09 18:30:39 +00:00
|
|
|
if (_mouseClicked) {
|
2013-01-21 11:34:50 +00:00
|
|
|
if (_klaymen) {
|
2012-10-09 18:30:39 +00:00
|
|
|
if (_canAcceptInput &&
|
2013-07-14 17:01:47 +00:00
|
|
|
_klaymen->hasMessageHandler() &&
|
2013-01-21 11:34:50 +00:00
|
|
|
sendMessage(_klaymen, 0x1008, 0) != 0 &&
|
2012-10-09 18:30:39 +00:00
|
|
|
queryPositionSprite(_mouseClickPos.x, _mouseClickPos.y)) {
|
2011-06-30 09:55:59 +00:00
|
|
|
_mouseClicked = false;
|
2012-10-09 18:30:39 +00:00
|
|
|
} else if (_canAcceptInput &&
|
2013-07-14 17:01:47 +00:00
|
|
|
_klaymen->hasMessageHandler() &&
|
2013-01-21 11:34:50 +00:00
|
|
|
sendMessage(_klaymen, 0x1008, 0) != 0) {
|
2012-10-09 18:30:39 +00:00
|
|
|
_mouseClicked = !queryPositionRectList(_mouseClickPos.x, _mouseClickPos.y);
|
2011-06-30 09:55:59 +00:00
|
|
|
}
|
2012-10-09 18:30:39 +00:00
|
|
|
} else if (queryPositionSprite(_mouseClickPos.x, _mouseClickPos.y)) {
|
|
|
|
_mouseClicked = false;
|
2011-06-30 09:55:59 +00:00
|
|
|
}
|
2012-10-09 18:30:39 +00:00
|
|
|
}
|
2011-07-08 19:09:33 +00:00
|
|
|
|
2012-10-09 18:30:39 +00:00
|
|
|
processMessageList();
|
2011-06-30 09:55:59 +00:00
|
|
|
|
2013-07-14 17:01:47 +00:00
|
|
|
// Update all entities
|
2012-10-09 18:30:39 +00:00
|
|
|
for (Common::Array<Entity*>::iterator iter = _entities.begin(); iter != _entities.end(); iter++)
|
|
|
|
(*iter)->handleUpdate();
|
2011-06-30 09:55:59 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-09-23 15:21:24 +00:00
|
|
|
void Scene::leaveScene(uint32 result) {
|
|
|
|
sendMessage(_parentModule, 0x1009, result);
|
|
|
|
}
|
|
|
|
|
2011-06-30 09:55:59 +00:00
|
|
|
uint32 Scene::handleMessage(int messageNum, const MessageParam ¶m, Entity *sender) {
|
|
|
|
switch (messageNum) {
|
2013-12-25 15:10:14 +00:00
|
|
|
case NM_MOUSE_MOVE:
|
2011-07-11 12:21:56 +00:00
|
|
|
if (_mouseCursor && _mouseCursor->hasMessageHandler())
|
2011-09-16 12:23:26 +00:00
|
|
|
sendMessage(_mouseCursor, 0x4002, param);
|
2011-06-30 09:55:59 +00:00
|
|
|
break;
|
2013-12-25 15:10:14 +00:00
|
|
|
case NM_MOUSE_CLICK:
|
2011-06-30 09:55:59 +00:00
|
|
|
_mouseClicked = true;
|
2011-07-08 19:09:33 +00:00
|
|
|
_mouseClickPos = param.asPoint();
|
2011-06-30 09:55:59 +00:00
|
|
|
break;
|
2013-01-19 22:37:51 +00:00
|
|
|
case 0x0006:
|
2013-07-14 17:01:47 +00:00
|
|
|
sendMessage(_parentModule, 0x1009, param);
|
2011-06-30 09:55:59 +00:00
|
|
|
break;
|
|
|
|
case 0x1006:
|
2013-01-21 11:34:50 +00:00
|
|
|
// Sent by Klaymen when its animation sequence has finished
|
|
|
|
if (_isKlaymenBusy) {
|
|
|
|
_isKlaymenBusy = false;
|
2012-10-09 18:30:39 +00:00
|
|
|
if (_messageListIndex == _messageListCount) {
|
|
|
|
// If the current message list was processed completely,
|
2013-01-21 11:34:50 +00:00
|
|
|
// sent Klaymen into the idle state.
|
2013-12-25 15:10:14 +00:00
|
|
|
sendMessage(_klaymen, NM_KLAYMEN_STAND_IDLE, 0);
|
2012-10-09 18:30:39 +00:00
|
|
|
} else {
|
|
|
|
// Else continue with the next message in the current message list
|
|
|
|
processMessageList();
|
2011-07-07 20:26:12 +00:00
|
|
|
}
|
2011-06-30 09:55:59 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 0x1007:
|
2012-10-09 18:30:39 +00:00
|
|
|
// This isn't sent by any code, check if it's in a message list
|
2013-01-21 11:34:50 +00:00
|
|
|
// This cancels the current message list and sets Klaymen into the idle state.
|
|
|
|
if (_isKlaymenBusy) {
|
|
|
|
_isKlaymenBusy = false;
|
2011-06-30 09:55:59 +00:00
|
|
|
_messageList = NULL;
|
2013-12-25 15:10:14 +00:00
|
|
|
sendMessage(_klaymen, NM_KLAYMEN_STAND_IDLE, 0);
|
2011-06-30 09:55:59 +00:00
|
|
|
}
|
|
|
|
break;
|
2013-12-25 15:10:14 +00:00
|
|
|
case NM_MOUSE_HIDE:
|
2011-07-11 12:21:56 +00:00
|
|
|
if (_mouseCursor) {
|
2012-10-09 18:30:39 +00:00
|
|
|
_mouseCursorWasVisible = _mouseCursor->getSurface()->getVisible();
|
2011-07-11 12:21:56 +00:00
|
|
|
_mouseCursor->getSurface()->setVisible(false);
|
2011-06-30 09:55:59 +00:00
|
|
|
}
|
|
|
|
break;
|
2013-12-25 15:10:14 +00:00
|
|
|
case NM_MOUSE_SHOW:
|
2012-10-09 18:30:39 +00:00
|
|
|
if (_mouseCursorWasVisible && _mouseCursor) {
|
2013-01-17 15:24:13 +00:00
|
|
|
_mouseCursor->getSurface()->setVisible(true);
|
2011-06-30 09:55:59 +00:00
|
|
|
}
|
|
|
|
break;
|
2013-12-25 15:10:14 +00:00
|
|
|
case NM_PRIORITY_CHANGE:
|
2012-10-09 18:30:39 +00:00
|
|
|
// Set the sender's surface priority
|
2011-07-08 19:09:33 +00:00
|
|
|
setSurfacePriority(((Sprite*)sender)->getSurface(), param.asInteger());
|
2011-06-30 09:55:59 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-07-08 19:09:33 +00:00
|
|
|
bool Scene::queryPositionSprite(int16 mouseX, int16 mouseY) {
|
2013-01-10 23:17:16 +00:00
|
|
|
for (uint i = 0; i < _collisionSprites.size(); i++) {
|
|
|
|
Sprite *sprite = _collisionSprites[i];
|
2013-07-14 17:01:47 +00:00
|
|
|
if (sprite->hasMessageHandler() && sprite->isPointInside(mouseX, mouseY) &&
|
2011-09-16 12:23:26 +00:00
|
|
|
sendPointMessage(sprite, 0x1011, _mouseClickPos) != 0) {
|
2011-07-08 19:09:33 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Scene::queryPositionRectList(int16 mouseX, int16 mouseY) {
|
2013-01-21 11:34:50 +00:00
|
|
|
int16 klaymenX = _klaymen->getX();
|
|
|
|
int16 klaymenY = _klaymen->getY();
|
2011-07-08 19:09:33 +00:00
|
|
|
if (_rectType == 1) {
|
|
|
|
RectList &rectList = *_rectList;
|
|
|
|
for (uint i = 0; i < rectList.size(); i++) {
|
2013-01-21 11:34:50 +00:00
|
|
|
debug(2, "(%d, %d) ? (%d, %d, %d, %d)", klaymenX, klaymenY, rectList[i].rect.x1, rectList[i].rect.y1, rectList[i].rect.x2, rectList[i].rect.y2);
|
|
|
|
if (rectList[i].rect.contains(klaymenX, klaymenY)) {
|
2011-07-08 19:09:33 +00:00
|
|
|
for (uint j = 0; j < rectList[i].subRects.size(); j++) {
|
2012-05-11 11:03:40 +00:00
|
|
|
debug(2, " (%d, %d) ? (%d, %d, %d, %d)", mouseX, mouseY, rectList[i].subRects[j].rect.x1, rectList[i].subRects[j].rect.y1, rectList[i].subRects[j].rect.x2, rectList[i].subRects[j].rect.y2);
|
|
|
|
if (rectList[i].subRects[j].rect.contains(mouseX, mouseY)) {
|
2011-07-08 19:09:33 +00:00
|
|
|
return setMessageList2(rectList[i].subRects[j].messageListId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-08-29 10:21:30 +00:00
|
|
|
} else if (_rectType == 2) {
|
2013-01-21 11:34:50 +00:00
|
|
|
MessageList *messageList = _dataResource.getMessageListAtPos(klaymenX, klaymenY, mouseX, mouseY);
|
2011-08-29 10:21:30 +00:00
|
|
|
if (messageList && messageList->size())
|
|
|
|
setMessageList2(messageList, true, true);
|
2011-07-08 19:09:33 +00:00
|
|
|
}
|
2011-07-09 18:45:51 +00:00
|
|
|
return true;
|
2011-07-08 19:09:33 +00:00
|
|
|
}
|
|
|
|
|
2012-11-14 12:21:13 +00:00
|
|
|
void Scene::setMessageList(uint32 id, bool canAcceptInput, bool doConvertMessages) {
|
|
|
|
setMessageList(_vm->_staticData->getMessageList(id), canAcceptInput, doConvertMessages);
|
2011-07-08 19:09:33 +00:00
|
|
|
}
|
|
|
|
|
2012-11-14 12:21:13 +00:00
|
|
|
void Scene::setMessageList(MessageList *messageList, bool canAcceptInput, bool doConvertMessages) {
|
2011-07-08 19:09:33 +00:00
|
|
|
_messageList = messageList;
|
|
|
|
_messageListCount = _messageList ? _messageList->size() : 0;
|
|
|
|
_messageListIndex = 0;
|
2013-01-21 11:34:50 +00:00
|
|
|
_isKlaymenBusy = false;
|
2012-10-09 18:30:39 +00:00
|
|
|
_canAcceptInput = canAcceptInput;
|
2012-11-14 12:21:13 +00:00
|
|
|
_doConvertMessages = doConvertMessages;
|
2011-07-08 19:09:33 +00:00
|
|
|
_messageListStatus = 1;
|
2013-01-21 11:34:50 +00:00
|
|
|
sendMessage(_klaymen, 0x101C, 0);
|
2011-07-08 19:09:33 +00:00
|
|
|
}
|
|
|
|
|
2012-11-14 12:21:13 +00:00
|
|
|
bool Scene::setMessageList2(uint32 id, bool canAcceptInput, bool doConvertMessages) {
|
|
|
|
return setMessageList2(_vm->_staticData->getMessageList(id), canAcceptInput, doConvertMessages);
|
2011-07-08 19:09:33 +00:00
|
|
|
}
|
|
|
|
|
2012-11-14 12:21:13 +00:00
|
|
|
bool Scene::setMessageList2(MessageList *messageList, bool canAcceptInput, bool doConvertMessages) {
|
2012-10-09 18:30:39 +00:00
|
|
|
if (_messageListStatus == 0 ||
|
|
|
|
(_messageListStatus == 1 && messageList != _messageList2) ||
|
|
|
|
(_messageListStatus == 2 && messageList == _messageList2)) {
|
|
|
|
// NOTE Skipped unneeded resource preloading code
|
|
|
|
_messageValue = -1;
|
2011-07-08 19:09:33 +00:00
|
|
|
_messageList2 = messageList;
|
2012-11-14 12:21:13 +00:00
|
|
|
setMessageList(messageList, canAcceptInput, doConvertMessages);
|
2012-10-09 18:30:39 +00:00
|
|
|
return true;
|
2011-07-08 19:09:33 +00:00
|
|
|
}
|
2012-10-09 18:30:39 +00:00
|
|
|
return false;
|
2011-07-08 19:09:33 +00:00
|
|
|
}
|
|
|
|
|
2013-01-17 10:04:17 +00:00
|
|
|
bool Scene::isMessageList2(uint32 id) {
|
|
|
|
return _messageList2 == _vm->_staticData->getMessageList(id);
|
|
|
|
}
|
|
|
|
|
2012-10-09 18:30:39 +00:00
|
|
|
void Scene::processMessageList() {
|
2013-01-21 11:34:50 +00:00
|
|
|
debug(7, "Scene::processMessageList() _isMessageListBusy = %d; _isKlaymenBusy = %d", _isMessageListBusy, _isKlaymenBusy);
|
2011-07-08 19:09:33 +00:00
|
|
|
|
2013-01-21 11:34:50 +00:00
|
|
|
if (_isMessageListBusy || _isKlaymenBusy)
|
2011-07-08 19:09:33 +00:00
|
|
|
return;
|
|
|
|
|
2012-10-09 18:30:39 +00:00
|
|
|
_isMessageListBusy = true;
|
2011-07-08 19:09:33 +00:00
|
|
|
|
|
|
|
if (!_messageList) {
|
|
|
|
_messageList2 = NULL;
|
|
|
|
_messageListStatus = 0;
|
|
|
|
}
|
2013-07-14 17:01:47 +00:00
|
|
|
|
|
|
|
if (_messageList && _klaymen) {
|
2013-01-18 19:00:28 +00:00
|
|
|
|
|
|
|
#if 0
|
|
|
|
debug("MessageList: %p, %d", (void*)_messageList, _messageList->size());
|
|
|
|
for (uint i = 0; i < _messageList->size(); ++i) {
|
|
|
|
if (i == _messageListIndex) debugN("**"); else debugN(" ");
|
|
|
|
debug("(%08X, %08X)", (*_messageList)[i].messageNum, (*_messageList)[i].messageValue);
|
|
|
|
}
|
|
|
|
debug("--------------------------------");
|
|
|
|
#endif
|
2013-07-14 17:01:47 +00:00
|
|
|
|
2013-01-21 11:34:50 +00:00
|
|
|
while (_messageList && _messageListIndex < _messageListCount && !_isKlaymenBusy) {
|
2011-08-29 10:21:30 +00:00
|
|
|
uint32 messageNum = (*_messageList)[_messageListIndex].messageNum;
|
2011-07-08 19:09:33 +00:00
|
|
|
uint32 messageParam = (*_messageList)[_messageListIndex].messageValue;
|
2013-07-14 17:01:47 +00:00
|
|
|
|
2012-11-14 12:21:13 +00:00
|
|
|
++_messageListIndex;
|
|
|
|
if (_messageListIndex == _messageListCount)
|
2013-01-21 11:34:50 +00:00
|
|
|
sendMessage(_klaymen, 0x1021, 0);
|
2012-11-14 12:21:13 +00:00
|
|
|
if (_doConvertMessages)
|
2011-08-29 10:21:30 +00:00
|
|
|
messageNum = convertMessageNum(messageNum);
|
2012-11-14 12:21:13 +00:00
|
|
|
if (messageNum == 0x1009 || messageNum == 0x1024) {
|
|
|
|
sendMessage(_parentModule, messageNum, messageParam);
|
|
|
|
} else if (messageNum == 0x100A) {
|
|
|
|
_messageValue = messageParam;
|
|
|
|
sendMessage(_parentModule, messageNum, messageParam);
|
|
|
|
} else if (messageNum == 0x4001) {
|
2013-01-21 11:34:50 +00:00
|
|
|
_isKlaymenBusy = true;
|
|
|
|
sendPointMessage(_klaymen, 0x4001, _mouseClickPos);
|
2012-11-14 12:21:13 +00:00
|
|
|
} else if (messageNum == 0x100D) {
|
2013-12-23 19:03:35 +00:00
|
|
|
if (this->hasMessageHandler() && sendMessage(this, NM_ANIMATION_START, messageParam) != 0)
|
2012-11-14 12:21:13 +00:00
|
|
|
continue;
|
|
|
|
} else if (messageNum == 0x101A) {
|
|
|
|
_messageListStatus = 0;
|
|
|
|
} else if (messageNum == 0x101B) {
|
|
|
|
_messageListStatus = 2;
|
|
|
|
} else if (messageNum == 0x1020) {
|
|
|
|
_canAcceptInput = false;
|
|
|
|
} else if (messageNum >= 0x2000 && messageNum <= 0x2FFF) {
|
|
|
|
if (this->hasMessageHandler() && sendMessage(this, messageNum, messageParam) != 0) {
|
|
|
|
_isMessageListBusy = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else if (messageNum != 0x4003) {
|
2013-01-21 11:34:50 +00:00
|
|
|
_isKlaymenBusy = true;
|
|
|
|
if (_klaymen->hasMessageHandler() && sendMessage(_klaymen, messageNum, messageParam) != 0) {
|
|
|
|
_isKlaymenBusy = false;
|
2012-11-14 12:21:13 +00:00
|
|
|
}
|
2013-07-14 17:01:47 +00:00
|
|
|
}
|
2011-07-08 19:09:33 +00:00
|
|
|
if (_messageListIndex == _messageListCount) {
|
2012-10-09 18:30:39 +00:00
|
|
|
_canAcceptInput = true;
|
2011-07-08 19:09:33 +00:00
|
|
|
_messageList = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-09 18:30:39 +00:00
|
|
|
_isMessageListBusy = false;
|
2013-07-14 17:01:47 +00:00
|
|
|
|
2011-07-08 19:09:33 +00:00
|
|
|
}
|
|
|
|
|
2012-10-09 18:30:39 +00:00
|
|
|
void Scene::cancelMessageList() {
|
2013-01-21 11:34:50 +00:00
|
|
|
_isKlaymenBusy = false;
|
2011-07-18 18:54:43 +00:00
|
|
|
_messageList = NULL;
|
2012-10-09 18:30:39 +00:00
|
|
|
_canAcceptInput = true;
|
2013-12-25 15:10:14 +00:00
|
|
|
sendMessage(_klaymen, NM_KLAYMEN_STAND_IDLE, 0);
|
2011-07-18 18:54:43 +00:00
|
|
|
}
|
|
|
|
|
2011-07-08 19:09:33 +00:00
|
|
|
void Scene::setRectList(uint32 id) {
|
|
|
|
setRectList(_vm->_staticData->getRectList(id));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Scene::setRectList(RectList *rectList) {
|
|
|
|
_rectList = rectList;
|
|
|
|
_rectType = 1;
|
|
|
|
}
|
|
|
|
|
2011-07-26 08:38:19 +00:00
|
|
|
void Scene::clearRectList() {
|
|
|
|
_rectList = NULL;
|
2011-07-27 18:02:30 +00:00
|
|
|
_rectType = 0;
|
2011-07-26 08:38:19 +00:00
|
|
|
}
|
|
|
|
|
2011-08-29 10:21:30 +00:00
|
|
|
void Scene::loadHitRectList() {
|
|
|
|
HitRectList *hitRectList = _dataResource.getHitRectList();
|
|
|
|
if (hitRectList) {
|
|
|
|
_hitRectList = *hitRectList;
|
2013-01-10 23:17:16 +00:00
|
|
|
setHitRects(&_hitRectList);
|
2011-08-29 10:21:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-07-16 17:27:21 +00:00
|
|
|
void Scene::loadDataResource(uint32 fileHash) {
|
|
|
|
_dataResource.load(fileHash);
|
2011-08-29 10:21:30 +00:00
|
|
|
_rectType = 2;
|
2013-01-21 11:34:50 +00:00
|
|
|
if (_klaymen)
|
|
|
|
_klaymen->loadDataResource(fileHash);
|
2011-07-16 17:27:21 +00:00
|
|
|
}
|
|
|
|
|
2011-08-29 10:21:30 +00:00
|
|
|
uint16 Scene::convertMessageNum(uint32 messageNum) {
|
|
|
|
switch (messageNum) {
|
|
|
|
case 0x00004004:
|
|
|
|
return 0x4001;
|
|
|
|
case 0x00000083:
|
|
|
|
return 0x100A;
|
|
|
|
case 0x044001C8:
|
|
|
|
return 0x481C;
|
|
|
|
case 0x02420480:
|
|
|
|
return 0x4818;
|
|
|
|
case 0x08004025:
|
|
|
|
return 0x100D;
|
|
|
|
case 0x04404281:
|
|
|
|
return 0x4824;
|
|
|
|
case 0x08400880:
|
|
|
|
return 0x4825;
|
|
|
|
case 0x08209081:
|
|
|
|
return 0x4823;
|
|
|
|
case 0x24000060:
|
|
|
|
return 0x1009;
|
|
|
|
case 0x42002200:
|
|
|
|
return 0x4004;
|
|
|
|
case 0x428D4894:
|
2013-07-14 17:01:47 +00:00
|
|
|
return 0x101A;
|
2011-08-29 10:21:30 +00:00
|
|
|
}
|
|
|
|
return 0x1000;
|
|
|
|
}
|
|
|
|
|
2013-01-10 23:17:16 +00:00
|
|
|
void Scene::clearHitRects() {
|
|
|
|
_hitRects = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
HitRect *Scene::findHitRectAtPos(int16 x, int16 y) {
|
|
|
|
static HitRect kDefaultHitRect = {NRect(), 0x5000};
|
|
|
|
if (_hitRects)
|
|
|
|
for (HitRectList::iterator it = _hitRects->begin(); it != _hitRects->end(); it++)
|
|
|
|
if ((*it).rect.contains(x, y))
|
|
|
|
return &(*it);
|
2013-07-14 17:01:47 +00:00
|
|
|
return &kDefaultHitRect;
|
2013-01-10 23:17:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Scene::addCollisionSprite(Sprite *sprite) {
|
|
|
|
int index = 0, insertIndex = -1;
|
|
|
|
for (Common::Array<Sprite*>::iterator iter = _collisionSprites.begin(); iter != _collisionSprites.end(); iter++) {
|
|
|
|
if ((*iter)->getPriority() > sprite->getPriority()) {
|
|
|
|
insertIndex = index;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
index++;
|
|
|
|
}
|
|
|
|
if (insertIndex >= 0)
|
|
|
|
_collisionSprites.insert_at(insertIndex, sprite);
|
|
|
|
else
|
2013-07-14 17:01:47 +00:00
|
|
|
_collisionSprites.push_back(sprite);
|
2013-01-10 23:17:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Scene::removeCollisionSprite(Sprite *sprite) {
|
|
|
|
for (uint index = 0; index < _collisionSprites.size(); index++) {
|
|
|
|
if (_collisionSprites[index] == sprite) {
|
|
|
|
_collisionSprites.remove_at(index);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Scene::clearCollisionSprites() {
|
|
|
|
_collisionSprites.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Scene::checkCollision(Sprite *sprite, uint16 flags, int messageNum, uint32 messageParam) {
|
|
|
|
for (Common::Array<Sprite*>::iterator iter = _collisionSprites.begin(); iter != _collisionSprites.end(); iter++) {
|
|
|
|
Sprite *collSprite = *iter;
|
|
|
|
if ((sprite->getFlags() & flags) && collSprite->checkCollision(sprite->getCollisionBounds())) {
|
|
|
|
sprite->sendMessage(collSprite, messageNum, messageParam);
|
|
|
|
}
|
2013-07-14 17:01:47 +00:00
|
|
|
}
|
2013-01-10 23:17:16 +00:00
|
|
|
}
|
|
|
|
|
2013-01-17 08:27:38 +00:00
|
|
|
void Scene::insertMouse(Mouse *mouseCursor) {
|
|
|
|
if (_mouseCursor)
|
|
|
|
deleteSprite((Sprite**)&_mouseCursor);
|
|
|
|
_mouseCursor = mouseCursor;
|
|
|
|
addEntity(_mouseCursor);
|
|
|
|
}
|
|
|
|
|
2013-09-14 09:28:09 +00:00
|
|
|
// StaticScene
|
|
|
|
|
|
|
|
StaticScene::StaticScene(NeverhoodEngine *vm, Module *parentModule, uint32 backgroundFileHash, uint32 cursorFileHash)
|
|
|
|
: Scene(vm, parentModule) {
|
|
|
|
|
|
|
|
SetMessageHandler(&StaticScene::handleMessage);
|
|
|
|
|
|
|
|
setBackground(backgroundFileHash);
|
|
|
|
setPalette(backgroundFileHash);
|
|
|
|
insertPuzzleMouse(cursorFileHash, 20, 620);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32 StaticScene::handleMessage(int messageNum, const MessageParam ¶m, Entity *sender) {
|
|
|
|
Scene::handleMessage(messageNum, param, sender);
|
|
|
|
switch (messageNum) {
|
2013-12-25 15:10:14 +00:00
|
|
|
case NM_MOUSE_CLICK:
|
2013-09-14 09:28:09 +00:00
|
|
|
if (param.asPoint().x <= 20 || param.asPoint().x >= 620)
|
|
|
|
leaveScene(0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-06-30 09:55:59 +00:00
|
|
|
} // End of namespace Neverhood
|