2004-04-12 21:40:49 +00:00
|
|
|
/* ScummVM - Scumm Interpreter
|
2005-01-01 16:20:17 +00:00
|
|
|
* Copyright (C) 2004-2005 The ScummVM project
|
2004-04-12 21:40:49 +00:00
|
|
|
*
|
|
|
|
* The ReInherit Engine is (C)2000-2003 by Daniel Balsom.
|
|
|
|
*
|
|
|
|
* 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
*
|
|
|
|
* $Header$
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2004-05-01 13:48:01 +00:00
|
|
|
// Scene management module
|
2004-08-02 16:20:35 +00:00
|
|
|
#include "saga/saga.h"
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-08-02 16:20:35 +00:00
|
|
|
#include "saga/gfx.h"
|
|
|
|
#include "saga/animation.h"
|
2004-08-10 18:31:33 +00:00
|
|
|
#include "saga/console.h"
|
2004-10-21 06:49:11 +00:00
|
|
|
#include "saga/interface.h"
|
2004-08-10 19:00:30 +00:00
|
|
|
#include "saga/events.h"
|
2004-08-02 16:20:35 +00:00
|
|
|
#include "saga/isomap.h"
|
|
|
|
#include "saga/objectmap.h"
|
2004-08-10 19:20:33 +00:00
|
|
|
#include "saga/palanim.h"
|
2005-05-23 02:23:34 +00:00
|
|
|
#include "saga/puzzle.h"
|
2004-08-02 16:20:35 +00:00
|
|
|
#include "saga/render.h"
|
|
|
|
#include "saga/script.h"
|
|
|
|
#include "saga/sound.h"
|
|
|
|
#include "saga/music.h"
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-08-02 16:20:35 +00:00
|
|
|
#include "saga/scene.h"
|
2004-12-15 00:24:12 +00:00
|
|
|
#include "saga/stream.h"
|
2004-12-18 20:33:02 +00:00
|
|
|
#include "saga/actor.h"
|
2005-07-19 19:05:52 +00:00
|
|
|
#include "saga/rscfile.h"
|
|
|
|
#include "saga/resnames.h"
|
2004-04-12 21:40:49 +00:00
|
|
|
|
|
|
|
namespace Saga {
|
|
|
|
|
2005-01-04 16:10:43 +00:00
|
|
|
static int initSceneDoors[SCENE_DOORS_MAX] = {
|
|
|
|
0, 0xff, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff
|
|
|
|
};
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
Scene::Scene(SagaEngine *vm) : _vm(vm) {
|
|
|
|
byte *sceneLUTPointer;
|
|
|
|
size_t sceneLUTLength;
|
|
|
|
uint32 resourceId;
|
2004-04-12 21:40:49 +00:00
|
|
|
int i;
|
|
|
|
|
|
|
|
|
2004-05-01 13:48:01 +00:00
|
|
|
// Load scene module resource context
|
2005-07-19 19:05:52 +00:00
|
|
|
_sceneContext = _vm->_resource->getContext(GAME_RESOURCEFILE);
|
2004-11-15 03:03:48 +00:00
|
|
|
if (_sceneContext == NULL) {
|
2005-07-19 19:05:52 +00:00
|
|
|
error("Scene::Scene() scene context not found");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-01 13:48:01 +00:00
|
|
|
// Load scene lookup table
|
2005-07-19 19:05:52 +00:00
|
|
|
resourceId = _vm->_resource->convertResourceId(_vm->getResourceDescription()->sceneLUTResourceId);
|
|
|
|
debug(3, "Loading scene LUT from resource %i", resourceId);
|
|
|
|
_vm->_resource->loadResource(_sceneContext, resourceId, sceneLUTPointer, sceneLUTLength);
|
|
|
|
if (sceneLUTLength == 0) {
|
|
|
|
error("Scene::Scene() sceneLUTLength == 0");
|
2004-12-15 00:24:12 +00:00
|
|
|
}
|
2005-07-19 19:05:52 +00:00
|
|
|
_sceneCount = sceneLUTLength / 2;
|
2005-07-08 18:39:53 +00:00
|
|
|
_sceneLUT = (int *)malloc(_sceneCount * sizeof(*_sceneLUT));
|
2004-08-04 20:28:57 +00:00
|
|
|
if (_sceneLUT == NULL) {
|
2005-01-21 21:55:54 +00:00
|
|
|
memoryError("Scene::Scene()");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
MemoryReadStreamEndian readS(sceneLUTPointer, sceneLUTLength, _sceneContext->isBigEndian);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-08 18:39:53 +00:00
|
|
|
for (i = 0; i < _sceneCount; i++) {
|
2004-12-15 00:24:12 +00:00
|
|
|
_sceneLUT[i] = readS.readUint16();
|
2005-05-08 14:04:06 +00:00
|
|
|
debug(8, "sceneNumber %i has resourceId %i", i, _sceneLUT[i]);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
free(sceneLUTPointer);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-01-09 23:41:22 +00:00
|
|
|
_firstScene = _vm->getStartSceneNumber();
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "First scene set to %d.", _firstScene);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-14 17:46:21 +00:00
|
|
|
debug(3, "LUT has %d entries.", _sceneCount);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-08-04 20:28:57 +00:00
|
|
|
_sceneLoaded = false;
|
|
|
|
_sceneNumber = 0;
|
2005-04-02 20:04:39 +00:00
|
|
|
_sceneResourceId = 0;
|
2004-08-04 20:28:57 +00:00
|
|
|
_inGame = false;
|
2005-05-08 14:04:06 +00:00
|
|
|
_loadDescription = false;
|
|
|
|
memset(&_sceneDescription, 0, sizeof(_sceneDescription));
|
2005-07-19 19:05:52 +00:00
|
|
|
_resourceListCount = 0;
|
|
|
|
_resourceList = NULL;
|
2004-08-04 20:28:57 +00:00
|
|
|
_sceneProc = NULL;
|
2005-01-18 11:55:31 +00:00
|
|
|
_objectMap = new ObjectMap(_vm);
|
|
|
|
_actionMap = new ObjectMap(_vm);
|
2004-08-04 20:28:57 +00:00
|
|
|
memset(&_bg, 0, sizeof(_bg));
|
|
|
|
memset(&_bgMask, 0, sizeof(_bgMask));
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-08-04 20:28:57 +00:00
|
|
|
Scene::~Scene() {
|
2005-07-19 19:05:52 +00:00
|
|
|
delete _actionMap;
|
|
|
|
delete _objectMap;
|
|
|
|
free(_sceneLUT);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-14 17:46:21 +00:00
|
|
|
void Scene::drawTextList(Surface *ds) {
|
|
|
|
TextListEntry *entry;
|
|
|
|
|
|
|
|
for (TextList::iterator textIterator = _textList.begin(); textIterator != _textList.end(); ++textIterator) {
|
|
|
|
entry = (TextListEntry *)textIterator.operator->();
|
2005-07-19 19:05:52 +00:00
|
|
|
if (entry->display) {
|
|
|
|
|
|
|
|
if (entry->useRect) {
|
|
|
|
_vm->_font->textDrawRect(entry->fontId, ds, entry->text, entry->rect, entry->color, entry->effectColor, entry->flags);
|
|
|
|
} else {
|
|
|
|
_vm->_font->textDraw(entry->fontId, ds, entry->text, entry->point, entry->color, entry->effectColor, entry->flags);
|
|
|
|
}
|
2005-07-14 17:46:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
void Scene::startScene() {
|
2004-12-15 00:24:12 +00:00
|
|
|
SceneQueueList::iterator queueIterator;
|
2005-05-08 14:04:06 +00:00
|
|
|
LoadSceneParams *sceneQueue;
|
2004-11-20 00:05:50 +00:00
|
|
|
EVENT event;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-08-04 20:28:57 +00:00
|
|
|
if (_sceneLoaded) {
|
2005-05-08 14:04:06 +00:00
|
|
|
error("Scene::start(): Error: Can't start game...scene already loaded");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-08-04 20:28:57 +00:00
|
|
|
if (_inGame) {
|
2005-05-08 14:04:06 +00:00
|
|
|
error("Scene::start(): Error: Can't start game...game already started");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-11-20 00:05:50 +00:00
|
|
|
// Hide cursor during intro
|
|
|
|
event.type = ONESHOT_EVENT;
|
|
|
|
event.code = CURSOR_EVENT;
|
|
|
|
event.op = EVENT_HIDE;
|
|
|
|
_vm->_events->queue(&event);
|
|
|
|
|
2005-01-11 21:10:36 +00:00
|
|
|
switch (_vm->getGameType()) {
|
2004-12-22 13:09:47 +00:00
|
|
|
case GType_ITE:
|
2004-08-11 23:42:02 +00:00
|
|
|
ITEStartProc();
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
2004-12-22 13:09:47 +00:00
|
|
|
case GType_IHNM:
|
2004-08-11 23:42:02 +00:00
|
|
|
IHNMStartProc();
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
|
|
|
default:
|
2005-05-08 14:04:06 +00:00
|
|
|
error("Scene::start(): Error: Can't start game... gametype not supported");
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2004-12-15 00:24:12 +00:00
|
|
|
// Load the head in scene queue
|
|
|
|
queueIterator = _sceneQueue.begin();
|
|
|
|
if (queueIterator == _sceneQueue.end()) {
|
2005-05-08 14:04:06 +00:00
|
|
|
return;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
sceneQueue = queueIterator.operator->();
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
loadScene(sceneQueue);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
void Scene::nextScene() {
|
2004-12-15 00:24:12 +00:00
|
|
|
SceneQueueList::iterator queueIterator;
|
2005-05-08 14:04:06 +00:00
|
|
|
LoadSceneParams *sceneQueue;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-08-04 20:28:57 +00:00
|
|
|
if (!_sceneLoaded) {
|
2005-05-08 14:04:06 +00:00
|
|
|
error("Scene::next(): Error: Can't advance scene...no scene loaded");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-08-04 20:28:57 +00:00
|
|
|
if (_inGame) {
|
2005-05-08 14:04:06 +00:00
|
|
|
error("Scene::next(): Error: Can't advance scene...game already started");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-08-04 20:28:57 +00:00
|
|
|
endScene();
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-12-15 00:24:12 +00:00
|
|
|
// Delete the current head in scene queue
|
|
|
|
queueIterator = _sceneQueue.begin();
|
|
|
|
if (queueIterator == _sceneQueue.end()) {
|
2005-05-08 14:04:06 +00:00
|
|
|
return;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
2004-12-15 00:24:12 +00:00
|
|
|
|
|
|
|
queueIterator = _sceneQueue.erase(queueIterator);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-12-15 00:24:12 +00:00
|
|
|
if (queueIterator == _sceneQueue.end()) {
|
2005-05-08 14:04:06 +00:00
|
|
|
return;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-12-15 00:24:12 +00:00
|
|
|
// Load the head in scene queue
|
2005-05-08 14:04:06 +00:00
|
|
|
sceneQueue = queueIterator.operator->();
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
loadScene(sceneQueue);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
void Scene::skipScene() {
|
2004-12-15 00:24:12 +00:00
|
|
|
SceneQueueList::iterator queueIterator;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
LoadSceneParams *sceneQueue = NULL;
|
|
|
|
LoadSceneParams *skipQueue = NULL;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-08-04 20:28:57 +00:00
|
|
|
if (!_sceneLoaded) {
|
2005-05-08 14:04:06 +00:00
|
|
|
error("Scene::skip(): Error: Can't skip scene...no scene loaded");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-08-04 20:28:57 +00:00
|
|
|
if (_inGame) {
|
2005-05-08 14:04:06 +00:00
|
|
|
error("Scene::skip(): Error: Can't skip scene...game already started");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-05-01 13:48:01 +00:00
|
|
|
// Walk down scene queue and try to find a skip target
|
2004-12-15 00:24:12 +00:00
|
|
|
queueIterator = _sceneQueue.begin();
|
|
|
|
if (queueIterator == _sceneQueue.end()) {
|
2005-05-08 14:04:06 +00:00
|
|
|
error("Scene::skip(): Error: Can't skip scene...no scenes in queue");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-12-15 00:24:12 +00:00
|
|
|
++queueIterator;
|
|
|
|
while (queueIterator != _sceneQueue.end()) {
|
2005-05-08 14:04:06 +00:00
|
|
|
sceneQueue = queueIterator.operator->();
|
|
|
|
assert(sceneQueue != NULL);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
if (sceneQueue->sceneSkipTarget) {
|
|
|
|
skipQueue = sceneQueue;
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
|
|
|
}
|
2004-12-15 00:24:12 +00:00
|
|
|
++queueIterator;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-05-01 13:48:01 +00:00
|
|
|
// If skip target found, remove preceding scenes and load
|
2005-05-08 14:04:06 +00:00
|
|
|
if (skipQueue != NULL) {
|
2004-12-15 00:24:12 +00:00
|
|
|
_sceneQueue.erase(_sceneQueue.begin(), queueIterator);
|
|
|
|
|
2004-08-04 20:28:57 +00:00
|
|
|
endScene();
|
2005-05-08 14:04:06 +00:00
|
|
|
loadScene(skipQueue);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
void Scene::changeScene(uint16 sceneNumber, int actorsEntrance, SceneTransitionType transitionType) {
|
|
|
|
LoadSceneParams sceneParams;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
sceneParams.actorsEntrance = actorsEntrance;
|
|
|
|
sceneParams.loadFlag = kLoadBySceneNumber;
|
|
|
|
sceneParams.sceneDescriptor = sceneNumber;
|
|
|
|
sceneParams.transitionType = transitionType;
|
|
|
|
sceneParams.sceneProc = NULL;
|
|
|
|
sceneParams.sceneSkipTarget = false;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-08-04 20:28:57 +00:00
|
|
|
endScene();
|
2005-05-08 14:04:06 +00:00
|
|
|
loadScene(&sceneParams);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2004-12-19 13:38:11 +00:00
|
|
|
void Scene::getSlopes(int &beginSlope, int &endSlope) {
|
2005-05-08 14:04:06 +00:00
|
|
|
beginSlope = _vm->getSceneHeight() - _sceneDescription.beginSlope;
|
|
|
|
endSlope = _vm->getSceneHeight() - _sceneDescription.endSlope;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-08 16:56:03 +00:00
|
|
|
void Scene::getBGInfo(BGInfo &bgInfo) {
|
|
|
|
bgInfo.buffer = _bg.buf;
|
|
|
|
bgInfo.bufferLength = _bg.buf_len;
|
|
|
|
bgInfo.bounds.left = 0;
|
|
|
|
bgInfo.bounds.top = 0;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-01-09 23:41:22 +00:00
|
|
|
if (_bg.w < _vm->getDisplayWidth()) {
|
2005-07-08 16:56:03 +00:00
|
|
|
bgInfo.bounds.left = (_vm->getDisplayWidth() - _bg.w) / 2;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-01-11 21:10:36 +00:00
|
|
|
if (_bg.h < _vm->getSceneHeight()) {
|
2005-07-09 10:34:09 +00:00
|
|
|
bgInfo.bounds.top = (_vm->getSceneHeight() - _bg.h) / 2;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-08 16:56:03 +00:00
|
|
|
bgInfo.bounds.setWidth(_bg.w);
|
|
|
|
bgInfo.bounds.setHeight(_bg.h);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-12-28 21:27:18 +00:00
|
|
|
int Scene::getBGMaskType(const Point &testPoint) {
|
|
|
|
uint offset;
|
|
|
|
if (!_bgMask.loaded) {
|
|
|
|
return 0;
|
|
|
|
}
|
2005-01-06 14:32:14 +00:00
|
|
|
offset = testPoint.x + testPoint.y * _bgMask.w;
|
2004-12-28 21:27:18 +00:00
|
|
|
if (offset >= _bgMask.buf_len) {
|
|
|
|
error("Scene::getBGMaskType offset 0x%X exceed bufferLength 0x%X", offset, _bgMask.buf_len);
|
|
|
|
}
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-12-28 21:27:18 +00:00
|
|
|
return (_bgMask.buf[offset] >> 4) & 0x0f;
|
|
|
|
}
|
|
|
|
|
2005-01-04 16:10:43 +00:00
|
|
|
bool Scene::validBGMaskPoint(const Point &testPoint) {
|
|
|
|
if (!_bgMask.loaded) {
|
|
|
|
error("Scene::validBGMaskPoint _bgMask not loaded");
|
|
|
|
}
|
|
|
|
|
|
|
|
return !((testPoint.x < 0) || (testPoint.x >= _bgMask.w) ||
|
|
|
|
(testPoint.y < 0) || (testPoint.y >= _bgMask.h));
|
|
|
|
}
|
|
|
|
|
2004-12-28 21:27:18 +00:00
|
|
|
bool Scene::canWalk(const Point &testPoint) {
|
|
|
|
int maskType;
|
2005-01-04 16:10:43 +00:00
|
|
|
|
2004-08-04 20:28:57 +00:00
|
|
|
if (!_bgMask.loaded) {
|
2004-12-28 21:27:18 +00:00
|
|
|
return true;
|
|
|
|
}
|
2005-01-04 16:10:43 +00:00
|
|
|
if (!validBGMaskPoint(testPoint)) {
|
2005-04-23 14:15:00 +00:00
|
|
|
return true;
|
2005-01-04 16:10:43 +00:00
|
|
|
}
|
|
|
|
|
2004-12-28 21:27:18 +00:00
|
|
|
maskType = getBGMaskType(testPoint);
|
|
|
|
return getDoorState(maskType) == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Scene::offscreenPath(Point &testPoint) {
|
2005-04-24 15:35:03 +00:00
|
|
|
Point point;
|
2004-12-28 21:27:18 +00:00
|
|
|
|
|
|
|
if (!_bgMask.loaded) {
|
|
|
|
return false;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-04-24 15:35:03 +00:00
|
|
|
point.x = clamp( 0, testPoint.x, _bgMask.w - 1 );
|
|
|
|
point.y = clamp( 0, testPoint.y, _bgMask.h - 1 );
|
|
|
|
if (point == testPoint) {
|
2004-12-28 21:27:18 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2005-04-24 15:35:03 +00:00
|
|
|
if (point.y >= _bgMask.h - 1) {
|
|
|
|
point.y = _bgMask.h - 2;
|
2004-12-28 21:27:18 +00:00
|
|
|
}
|
2005-04-24 15:35:03 +00:00
|
|
|
testPoint = point;
|
2004-12-28 21:27:18 +00:00
|
|
|
|
|
|
|
return true;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-12-28 21:27:18 +00:00
|
|
|
void Scene::getBGMaskInfo(int &width, int &height, byte *&buffer, size_t &bufferLength) {
|
|
|
|
if (!_bgMask.loaded) {
|
|
|
|
error("Scene::getBGMaskInfo _bgMask not loaded");
|
|
|
|
}
|
|
|
|
|
|
|
|
width = _bgMask.w;
|
|
|
|
height = _bgMask.h;
|
|
|
|
buffer = _bgMask.buf;
|
|
|
|
bufferLength = _bgMask.buf_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Scene::setDoorState(int doorNumber, int doorState) {
|
|
|
|
if ((doorNumber < 0) || (doorNumber >= SCENE_DOORS_MAX))
|
|
|
|
error("Scene::setDoorState wrong doorNumber");
|
|
|
|
|
|
|
|
_sceneDoors[doorNumber] = doorState;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Scene::getDoorState(int doorNumber) {
|
|
|
|
if ((doorNumber < 0) || (doorNumber >= SCENE_DOORS_MAX))
|
|
|
|
error("Scene::getDoorState wrong doorNumber");
|
|
|
|
|
|
|
|
return _sceneDoors[doorNumber];
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-01-04 16:10:43 +00:00
|
|
|
void Scene::initDoorsState() {
|
2005-01-04 18:54:29 +00:00
|
|
|
memcpy(_sceneDoors, initSceneDoors, sizeof (_sceneDoors) );
|
2005-01-04 16:10:43 +00:00
|
|
|
}
|
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
void Scene::loadScene(LoadSceneParams *loadSceneParams) {
|
2005-07-19 19:05:52 +00:00
|
|
|
size_t i;
|
2005-01-21 21:55:54 +00:00
|
|
|
EVENT event;
|
|
|
|
EVENT *q_event;
|
2005-07-09 16:23:45 +00:00
|
|
|
static PalEntry current_pal[PAL_ENTRIES];
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-08-04 20:28:57 +00:00
|
|
|
if (_sceneLoaded) {
|
2005-05-08 14:04:06 +00:00
|
|
|
error("Scene::loadScene(): Error, a scene is already loaded");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
_loadDescription = true;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
switch (loadSceneParams->loadFlag) {
|
|
|
|
case kLoadByResourceId:
|
|
|
|
_sceneNumber = 0; // original assign zero for loaded by resource id
|
|
|
|
_sceneResourceId = loadSceneParams->sceneDescriptor;
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
2005-05-08 14:04:06 +00:00
|
|
|
case kLoadBySceneNumber:
|
|
|
|
_sceneNumber = loadSceneParams->sceneDescriptor;
|
|
|
|
_sceneResourceId = getSceneResourceId(_sceneNumber);
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
2005-05-08 14:04:06 +00:00
|
|
|
case kLoadByDescription:
|
|
|
|
_sceneNumber = -1;
|
|
|
|
_sceneResourceId = -1;
|
|
|
|
assert(loadSceneParams->sceneDescription != NULL);
|
2005-07-19 19:05:52 +00:00
|
|
|
assert(loadSceneParams->sceneDescription->resourceList != NULL);
|
2005-05-08 14:04:06 +00:00
|
|
|
_loadDescription = false;
|
|
|
|
_sceneDescription = *loadSceneParams->sceneDescription;
|
2005-07-19 19:05:52 +00:00
|
|
|
_resourceList = loadSceneParams->sceneDescription->resourceList;
|
|
|
|
_resourceListCount = loadSceneParams->sceneDescription->resourceListCount;
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
|
|
|
}
|
2005-06-19 14:06:20 +00:00
|
|
|
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "Loading scene number %u:", _sceneNumber);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-05-01 13:48:01 +00:00
|
|
|
// Load scene descriptor and resource list resources
|
2005-05-08 14:04:06 +00:00
|
|
|
if (_loadDescription) {
|
2005-07-19 19:05:52 +00:00
|
|
|
debug(3, "Loading scene resource %i", _sceneResourceId);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
loadSceneDescriptor(_sceneResourceId);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
loadSceneResourceList(_sceneDescription.resourceListResourceId);
|
2004-04-12 21:40:49 +00:00
|
|
|
} else {
|
2005-07-19 19:05:52 +00:00
|
|
|
debug(3, "Loading memory scene resource");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-05-01 13:48:01 +00:00
|
|
|
// Load resources from scene resource list
|
2005-07-19 19:05:52 +00:00
|
|
|
for (i = 0; i < _resourceListCount; i++) {
|
|
|
|
if (!_resourceList[i].invalid) {
|
|
|
|
_vm->_resource->loadResource(_sceneContext, _resourceList[i].resourceId,
|
|
|
|
_resourceList[i].buffer, _resourceList[i].size);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-05-01 13:48:01 +00:00
|
|
|
// Process resources from scene resource list
|
2005-07-19 19:05:52 +00:00
|
|
|
processSceneResources();
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
if (_sceneDescription.flags & kSceneFlagISO) {
|
2005-04-21 01:30:07 +00:00
|
|
|
_outsetSceneNumber = _sceneNumber;
|
2005-07-08 16:56:03 +00:00
|
|
|
|
|
|
|
_sceneClip.left = 0;
|
|
|
|
_sceneClip.top = 0;
|
|
|
|
_sceneClip.right = _vm->getDisplayWidth();
|
2005-07-09 10:34:09 +00:00
|
|
|
_sceneClip.bottom = _vm->getSceneHeight();
|
2005-04-21 01:30:07 +00:00
|
|
|
} else {
|
2005-07-08 16:56:03 +00:00
|
|
|
BGInfo backGroundInfo;
|
|
|
|
getBGInfo(backGroundInfo);
|
|
|
|
_sceneClip = backGroundInfo.bounds;
|
2005-04-21 01:30:07 +00:00
|
|
|
if (!(_bg.w < _vm->getDisplayWidth() || _bg.h < _vm->getSceneHeight()))
|
|
|
|
_outsetSceneNumber = _sceneNumber;
|
|
|
|
}
|
|
|
|
|
2004-08-04 20:28:57 +00:00
|
|
|
_sceneLoaded = true;
|
2005-01-21 21:55:54 +00:00
|
|
|
|
|
|
|
q_event = NULL;
|
2005-06-16 16:46:50 +00:00
|
|
|
|
|
|
|
//fix placard bug
|
|
|
|
//i guess we should remove RF_PLACARD flag - and use _interface->getMode()
|
|
|
|
event.type = ONESHOT_EVENT;
|
|
|
|
event.code = GRAPHICS_EVENT;
|
|
|
|
event.op = EVENT_CLEARFLAG;
|
|
|
|
event.param = RF_PLACARD;
|
|
|
|
|
|
|
|
q_event = _vm->_events->chain(q_event, &event);
|
|
|
|
|
2005-07-08 18:39:53 +00:00
|
|
|
if (loadSceneParams->transitionType == kTransitionFade ) {
|
2004-08-11 22:27:39 +00:00
|
|
|
|
2005-06-04 15:02:17 +00:00
|
|
|
_vm->_interface->setFadeMode(kFadeOut);
|
2005-01-07 00:57:43 +00:00
|
|
|
|
2004-08-11 22:27:39 +00:00
|
|
|
// Fade to black out
|
|
|
|
_vm->_gfx->getCurrentPal(current_pal);
|
2004-10-27 21:32:28 +00:00
|
|
|
event.type = IMMEDIATE_EVENT;
|
|
|
|
event.code = PAL_EVENT;
|
2004-08-11 22:27:39 +00:00
|
|
|
event.op = EVENT_PALTOBLACK;
|
|
|
|
event.time = 0;
|
2004-12-29 19:40:59 +00:00
|
|
|
event.duration = kNormalFadeDuration;
|
2004-08-11 22:27:39 +00:00
|
|
|
event.data = current_pal;
|
|
|
|
q_event = _vm->_events->queue(&event);
|
|
|
|
|
2005-06-04 15:02:17 +00:00
|
|
|
// set fade mode
|
2004-10-27 21:32:28 +00:00
|
|
|
event.type = IMMEDIATE_EVENT;
|
2005-06-04 15:02:17 +00:00
|
|
|
event.code = INTERFACE_EVENT;
|
|
|
|
event.op = EVENT_SET_FADE_MODE;
|
|
|
|
event.param = kNoFade;
|
2004-08-11 22:27:39 +00:00
|
|
|
event.time = 0;
|
|
|
|
event.duration = 0;
|
|
|
|
q_event = _vm->_events->chain(q_event, &event);
|
|
|
|
|
2005-06-04 15:02:17 +00:00
|
|
|
// Display scene background, but stay with black palette
|
2005-01-07 00:57:43 +00:00
|
|
|
event.type = IMMEDIATE_EVENT;
|
2005-06-04 15:02:17 +00:00
|
|
|
event.code = BG_EVENT;
|
|
|
|
event.op = EVENT_DISPLAY;
|
|
|
|
event.param = NO_SET_PALETTE;
|
2005-01-07 00:57:43 +00:00
|
|
|
event.time = 0;
|
|
|
|
event.duration = 0;
|
|
|
|
q_event = _vm->_events->chain(q_event, &event);
|
2005-06-04 15:02:17 +00:00
|
|
|
|
2005-01-21 21:55:54 +00:00
|
|
|
}
|
2005-01-07 00:57:43 +00:00
|
|
|
|
2005-01-21 21:55:54 +00:00
|
|
|
// Start the scene pre script, but stay with black palette
|
2005-05-08 14:04:06 +00:00
|
|
|
if (_sceneDescription.startScriptEntrypointNumber > 0) {
|
2005-01-21 21:55:54 +00:00
|
|
|
event.type = ONESHOT_EVENT;
|
|
|
|
event.code = SCRIPT_EVENT;
|
|
|
|
event.op = EVENT_EXEC_BLOCKING;
|
|
|
|
event.time = 0;
|
2005-05-08 14:04:06 +00:00
|
|
|
event.param = _sceneDescription.scriptModuleNumber;
|
|
|
|
event.param2 = _sceneDescription.startScriptEntrypointNumber;
|
2005-01-21 21:55:54 +00:00
|
|
|
event.param3 = 0; // Action
|
|
|
|
event.param4 = _sceneNumber; // Object
|
2005-05-08 14:04:06 +00:00
|
|
|
event.param5 = loadSceneParams->actorsEntrance; // With Object
|
2005-01-21 21:55:54 +00:00
|
|
|
event.param6 = 0; // Actor
|
2004-10-16 07:46:52 +00:00
|
|
|
|
2005-01-21 21:55:54 +00:00
|
|
|
q_event = _vm->_events->chain(q_event, &event);
|
|
|
|
}
|
2004-10-16 07:46:52 +00:00
|
|
|
|
2005-07-08 18:39:53 +00:00
|
|
|
if (loadSceneParams->transitionType == kTransitionFade) {
|
2005-06-04 15:02:17 +00:00
|
|
|
|
|
|
|
// set fade mode
|
|
|
|
event.type = IMMEDIATE_EVENT;
|
|
|
|
event.code = INTERFACE_EVENT;
|
|
|
|
event.op = EVENT_SET_FADE_MODE;
|
|
|
|
event.param = kFadeIn;
|
|
|
|
event.time = 0;
|
|
|
|
event.duration = 0;
|
|
|
|
q_event = _vm->_events->chain(q_event, &event);
|
|
|
|
|
2004-08-11 22:27:39 +00:00
|
|
|
// Fade in from black to the scene background palette
|
2004-10-27 21:32:28 +00:00
|
|
|
event.type = IMMEDIATE_EVENT;
|
|
|
|
event.code = PAL_EVENT;
|
2004-08-11 22:27:39 +00:00
|
|
|
event.op = EVENT_BLACKTOPAL;
|
|
|
|
event.time = 0;
|
2004-12-29 19:40:59 +00:00
|
|
|
event.duration = kNormalFadeDuration;
|
2004-08-11 22:27:39 +00:00
|
|
|
event.data = _bg.pal;
|
|
|
|
|
|
|
|
q_event = _vm->_events->chain(q_event, &event);
|
2005-06-04 15:02:17 +00:00
|
|
|
|
|
|
|
// set fade mode
|
|
|
|
event.type = IMMEDIATE_EVENT;
|
|
|
|
event.code = INTERFACE_EVENT;
|
|
|
|
event.op = EVENT_SET_FADE_MODE;
|
|
|
|
event.param = kNoFade;
|
|
|
|
event.time = 0;
|
|
|
|
event.duration = 0;
|
|
|
|
q_event = _vm->_events->chain(q_event, &event);
|
2004-08-11 22:27:39 +00:00
|
|
|
}
|
|
|
|
|
2005-03-29 22:14:20 +00:00
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
if (loadSceneParams->sceneProc == NULL) {
|
2005-03-29 22:14:20 +00:00
|
|
|
if (!_inGame) {
|
|
|
|
_inGame = true;
|
2005-05-22 11:59:22 +00:00
|
|
|
_vm->_interface->setMode(kPanelMain);
|
2005-03-29 22:14:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
_vm->_sound->stopVoice();
|
|
|
|
_vm->_sound->stopSound();
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
if (_sceneDescription.musicResourceId >= 0) {
|
2005-03-29 22:14:20 +00:00
|
|
|
event.type = ONESHOT_EVENT;
|
|
|
|
event.code = MUSIC_EVENT;
|
2005-07-19 19:05:52 +00:00
|
|
|
event.param = _sceneDescription.musicResourceId;
|
2005-03-29 22:14:20 +00:00
|
|
|
event.param2 = MUSIC_DEFAULT;
|
|
|
|
event.op = EVENT_PLAY;
|
|
|
|
event.time = 0;
|
|
|
|
|
|
|
|
_vm->_events->queue(&event);
|
|
|
|
} else {
|
|
|
|
event.type = ONESHOT_EVENT;
|
|
|
|
event.code = MUSIC_EVENT;
|
|
|
|
event.op = EVENT_STOP;
|
|
|
|
event.time = 0;
|
|
|
|
|
|
|
|
_vm->_events->queue(&event);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set scene background
|
|
|
|
event.type = ONESHOT_EVENT;
|
|
|
|
event.code = BG_EVENT;
|
|
|
|
event.op = EVENT_DISPLAY;
|
|
|
|
event.param = SET_PALETTE;
|
|
|
|
event.time = 0;
|
|
|
|
|
|
|
|
_vm->_events->queue(&event);
|
|
|
|
|
2005-07-06 17:51:41 +00:00
|
|
|
if (getFlags() & kSceneFlagShowCursor) {
|
|
|
|
// Activate user interface
|
|
|
|
event.type = ONESHOT_EVENT;
|
|
|
|
event.code = INTERFACE_EVENT;
|
|
|
|
event.op = EVENT_ACTIVATE;
|
|
|
|
event.time = 0;
|
2005-03-29 22:14:20 +00:00
|
|
|
|
2005-07-06 17:51:41 +00:00
|
|
|
_vm->_events->queue(&event);
|
|
|
|
}
|
2005-03-29 22:14:20 +00:00
|
|
|
|
|
|
|
// Begin palette cycle animation if present
|
|
|
|
event.type = ONESHOT_EVENT;
|
|
|
|
event.code = PALANIM_EVENT;
|
|
|
|
event.op = EVENT_CYCLESTART;
|
|
|
|
event.time = 0;
|
|
|
|
|
|
|
|
q_event = _vm->_events->queue(&event);
|
|
|
|
|
|
|
|
// Show cursor
|
|
|
|
event.type = ONESHOT_EVENT;
|
|
|
|
event.code = CURSOR_EVENT;
|
|
|
|
event.op = EVENT_SHOW;
|
|
|
|
_vm->_events->chain(q_event, &event);
|
|
|
|
|
|
|
|
// Start the scene main script
|
2005-05-08 14:04:06 +00:00
|
|
|
if (_sceneDescription.sceneScriptEntrypointNumber > 0) {
|
2005-03-29 22:14:20 +00:00
|
|
|
event.type = ONESHOT_EVENT;
|
|
|
|
event.code = SCRIPT_EVENT;
|
|
|
|
event.op = EVENT_EXEC_NONBLOCKING;
|
|
|
|
event.time = 0;
|
2005-05-08 14:04:06 +00:00
|
|
|
event.param = _sceneDescription.scriptModuleNumber;
|
|
|
|
event.param2 = _sceneDescription.sceneScriptEntrypointNumber;
|
2005-03-29 22:14:20 +00:00
|
|
|
event.param3 = kVerbEnter; // Action
|
|
|
|
event.param4 = _sceneNumber; // Object
|
2005-05-08 14:04:06 +00:00
|
|
|
event.param5 = loadSceneParams->actorsEntrance; // With Object
|
2005-03-29 22:14:20 +00:00
|
|
|
event.param6 = 0; // Actor
|
|
|
|
|
|
|
|
_vm->_events->queue(&event);
|
|
|
|
}
|
|
|
|
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "Scene started");
|
2005-03-29 22:14:20 +00:00
|
|
|
|
2004-04-12 21:40:49 +00:00
|
|
|
} else {
|
2005-05-08 14:04:06 +00:00
|
|
|
loadSceneParams->sceneProc(SCENE_BEGIN, this);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-02-24 08:03:47 +00:00
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
// We probably don't want "followers" to go into scene -1 , 0. At the very
|
2005-02-24 08:03:47 +00:00
|
|
|
// least we don't want garbage to be drawn that early in the ITE intro.
|
2005-05-23 02:23:34 +00:00
|
|
|
if (_sceneNumber > 0 && _sceneNumber != ITE_SCENE_PUZZLE)
|
2005-05-08 14:04:06 +00:00
|
|
|
_vm->_actor->updateActorsScene(loadSceneParams->actorsEntrance);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-05-23 02:23:34 +00:00
|
|
|
if (_sceneNumber == ITE_SCENE_PUZZLE)
|
|
|
|
_vm->_puzzle->execute();
|
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
if (_sceneDescription.flags & kSceneFlagShowCursor)
|
2005-01-04 17:15:53 +00:00
|
|
|
_vm->_interface->activate();
|
|
|
|
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
void Scene::loadSceneDescriptor(uint32 resourceId) {
|
|
|
|
byte *sceneDescriptorData;
|
|
|
|
size_t sceneDescriptorDataLength;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
_vm->_resource->loadResource(_sceneContext, resourceId, sceneDescriptorData, sceneDescriptorDataLength);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
MemoryReadStreamEndian readS(sceneDescriptorData, sceneDescriptorDataLength, _sceneContext->isBigEndian);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
_sceneDescription.flags = readS.readSint16();
|
2005-07-19 19:05:52 +00:00
|
|
|
_sceneDescription.resourceListResourceId = readS.readSint16();
|
2005-05-08 14:04:06 +00:00
|
|
|
_sceneDescription.endSlope = readS.readSint16();
|
|
|
|
_sceneDescription.beginSlope = readS.readSint16();
|
|
|
|
_sceneDescription.scriptModuleNumber = readS.readUint16();
|
|
|
|
_sceneDescription.sceneScriptEntrypointNumber = readS.readUint16();
|
|
|
|
_sceneDescription.startScriptEntrypointNumber = readS.readUint16();
|
2005-07-19 19:05:52 +00:00
|
|
|
_sceneDescription.musicResourceId = readS.readSint16();
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
free(sceneDescriptorData);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
void Scene::loadSceneResourceList(uint32 resourceId) {
|
|
|
|
byte *resourceListData;
|
|
|
|
size_t resourceListDataLength;
|
|
|
|
size_t i;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-05-01 13:48:01 +00:00
|
|
|
// Load the scene resource table
|
2005-07-19 19:05:52 +00:00
|
|
|
_vm->_resource->loadResource(_sceneContext, resourceId, resourceListData, resourceListDataLength);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
MemoryReadStreamEndian readS(resourceListData, resourceListDataLength, _sceneContext->isBigEndian);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-05-01 13:48:01 +00:00
|
|
|
// Allocate memory for scene resource list
|
2005-07-19 19:05:52 +00:00
|
|
|
_resourceListCount = resourceListDataLength / SAGA_RESLIST_ENTRY_LEN;
|
|
|
|
debug(3, "Scene resource list contains %i entries", _resourceListCount);
|
|
|
|
_resourceList = (SceneResourceData *)calloc(_resourceListCount, sizeof(*_resourceList));
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-05-01 13:48:01 +00:00
|
|
|
// Load scene resource list from raw scene
|
|
|
|
// resource table
|
2005-07-19 19:05:52 +00:00
|
|
|
debug(3, "Loading scene resource list");
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
for (i = 0; i < _resourceListCount; i++) {
|
|
|
|
_resourceList[i].resourceId = readS.readUint16();
|
|
|
|
_resourceList[i].reourceType = readS.readUint16();
|
|
|
|
// demo version may contain invalid resourceId
|
|
|
|
_resourceList[i].invalid = !_vm->_resource->validResourceId(_sceneContext, _resourceList[i].resourceId);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
free(resourceListData);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
void Scene::processSceneResources() {
|
2005-06-11 17:24:06 +00:00
|
|
|
byte *resourceData;
|
2005-07-05 15:15:35 +00:00
|
|
|
uint16 resourceDataLength;
|
2005-07-19 19:05:52 +00:00
|
|
|
const byte *palPointer;
|
|
|
|
size_t i;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-05-01 13:48:01 +00:00
|
|
|
// Process the scene resource list
|
2005-07-19 19:05:52 +00:00
|
|
|
for (i = 0; i < _resourceListCount; i++) {
|
|
|
|
if (_resourceList[i].invalid) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
resourceData = _resourceList[i].buffer;
|
|
|
|
resourceDataLength = _resourceList[i].size;
|
|
|
|
switch (_resourceList[i].reourceType) {
|
|
|
|
case SAGA_ACTOR:
|
|
|
|
break;
|
|
|
|
case SAGA_OBJECT:
|
|
|
|
break;
|
2004-05-01 13:48:01 +00:00
|
|
|
case SAGA_BG_IMAGE: // Scene background resource
|
2004-08-04 20:28:57 +00:00
|
|
|
if (_bg.loaded) {
|
2005-07-19 19:05:52 +00:00
|
|
|
error("Scene::processSceneResources() Multiple background resources encountered");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "Loading background resource.");
|
2005-06-11 17:24:06 +00:00
|
|
|
_bg.res_buf = resourceData;
|
|
|
|
_bg.res_len = resourceDataLength;
|
2004-08-04 20:28:57 +00:00
|
|
|
_bg.loaded = 1;
|
|
|
|
|
|
|
|
if (_vm->decodeBGImage(_bg.res_buf,
|
|
|
|
_bg.res_len,
|
|
|
|
&_bg.buf,
|
|
|
|
&_bg.buf_len,
|
|
|
|
&_bg.w,
|
2004-10-27 21:32:28 +00:00
|
|
|
&_bg.h) != SUCCESS) {
|
2005-07-19 19:05:52 +00:00
|
|
|
error("Scene::processSceneResources() Error loading background resource %i", _resourceList[i].resourceId);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
palPointer = _vm->getImagePal(_bg.res_buf, _bg.res_len);
|
|
|
|
memcpy(_bg.pal, palPointer, sizeof(_bg.pal));
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
2004-05-01 13:48:01 +00:00
|
|
|
case SAGA_BG_MASK: // Scene background mask resource
|
2004-08-04 20:28:57 +00:00
|
|
|
if (_bgMask.loaded) {
|
2005-06-11 17:24:06 +00:00
|
|
|
error("Scene::ProcessSceneResources(): Duplicate background mask resource encountered");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "Loading BACKGROUND MASK resource.");
|
2005-06-11 17:24:06 +00:00
|
|
|
_bgMask.res_buf = resourceData;
|
|
|
|
_bgMask.res_len = resourceDataLength;
|
2004-08-04 20:28:57 +00:00
|
|
|
_bgMask.loaded = 1;
|
|
|
|
_vm->decodeBGImage(_bgMask.res_buf, _bgMask.res_len, &_bgMask.buf,
|
|
|
|
&_bgMask.buf_len, &_bgMask.w, &_bgMask.h);
|
2005-07-04 07:46:48 +00:00
|
|
|
|
|
|
|
// At least in ITE the mask needs to be clipped.
|
|
|
|
|
|
|
|
_bgMask.w = MIN(_bgMask.w, _vm->getDisplayWidth());
|
2005-07-09 10:34:09 +00:00
|
|
|
_bgMask.h = MIN(_bgMask.h, _vm->getSceneHeight());
|
2005-07-04 07:46:48 +00:00
|
|
|
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(4, "BACKGROUND MASK width=%d height=%d length=%d", _bgMask.w, _bgMask.h, _bgMask.buf_len);
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
2005-01-28 19:25:41 +00:00
|
|
|
case SAGA_STRINGS:
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "Loading scene strings resource...");
|
2005-06-11 17:24:06 +00:00
|
|
|
_vm->loadStrings(_sceneStrings, resourceData, resourceDataLength);
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
|
|
|
case SAGA_OBJECT_MAP:
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "Loading object map resource...");
|
2005-06-11 17:24:06 +00:00
|
|
|
_objectMap->load(resourceData, resourceDataLength);
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
|
|
|
case SAGA_ACTION_MAP:
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "Loading action map resource...");
|
2005-06-11 17:24:06 +00:00
|
|
|
_actionMap->load(resourceData, resourceDataLength);
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
2005-01-28 19:25:41 +00:00
|
|
|
case SAGA_ISO_IMAGES:
|
2005-05-08 14:04:06 +00:00
|
|
|
if (!(_sceneDescription.flags & kSceneFlagISO)) {
|
2005-01-28 19:25:41 +00:00
|
|
|
error("Scene::ProcessSceneResources(): not Iso mode");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "Loading isometric images resource.");
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-06-11 17:24:06 +00:00
|
|
|
_vm->_isoMap->loadImages(resourceData, resourceDataLength);
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
2005-01-28 19:25:41 +00:00
|
|
|
case SAGA_ISO_MAP:
|
2005-05-08 14:04:06 +00:00
|
|
|
if (!(_sceneDescription.flags & kSceneFlagISO)) {
|
2005-01-28 19:25:41 +00:00
|
|
|
error("Scene::ProcessSceneResources(): not Iso mode");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "Loading isometric map resource.");
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-06-11 17:24:06 +00:00
|
|
|
_vm->_isoMap->loadMap(resourceData, resourceDataLength);
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
2005-01-28 19:25:41 +00:00
|
|
|
case SAGA_ISO_PLATFORMS:
|
2005-05-08 14:04:06 +00:00
|
|
|
if (!(_sceneDescription.flags & kSceneFlagISO)) {
|
2005-01-28 19:25:41 +00:00
|
|
|
error("Scene::ProcessSceneResources(): not Iso mode");
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "Loading isometric platforms resource.");
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-06-11 17:24:06 +00:00
|
|
|
_vm->_isoMap->loadPlatforms(resourceData, resourceDataLength);
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
2005-01-28 19:25:41 +00:00
|
|
|
case SAGA_ISO_METATILES:
|
2005-05-08 14:04:06 +00:00
|
|
|
if (!(_sceneDescription.flags & kSceneFlagISO)) {
|
2005-01-28 19:25:41 +00:00
|
|
|
error("Scene::ProcessSceneResources(): not Iso mode");
|
|
|
|
}
|
|
|
|
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "Loading isometric metatiles resource.");
|
2005-01-28 19:25:41 +00:00
|
|
|
|
2005-06-11 17:24:06 +00:00
|
|
|
_vm->_isoMap->loadMetaTiles(resourceData, resourceDataLength);
|
2005-01-28 19:25:41 +00:00
|
|
|
break;
|
2004-04-12 21:40:49 +00:00
|
|
|
case SAGA_ANIM_1:
|
|
|
|
case SAGA_ANIM_2:
|
|
|
|
case SAGA_ANIM_3:
|
|
|
|
case SAGA_ANIM_4:
|
|
|
|
case SAGA_ANIM_5:
|
|
|
|
case SAGA_ANIM_6:
|
|
|
|
case SAGA_ANIM_7:
|
|
|
|
{
|
2005-07-19 19:05:52 +00:00
|
|
|
uint16 animId = _resourceList[i].reourceType - SAGA_ANIM_1;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "Loading animation resource animId=%i", animId);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-06-20 11:46:34 +00:00
|
|
|
_vm->_anim->load(animId, resourceData, resourceDataLength);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
break;
|
2005-01-28 19:25:41 +00:00
|
|
|
case SAGA_ISO_MULTI:
|
2005-05-08 14:04:06 +00:00
|
|
|
if (!(_sceneDescription.flags & kSceneFlagISO)) {
|
2005-01-28 19:25:41 +00:00
|
|
|
error("Scene::ProcessSceneResources(): not Iso mode");
|
|
|
|
}
|
|
|
|
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "Loading isometric multi resource.");
|
2005-01-28 19:25:41 +00:00
|
|
|
|
2005-06-11 17:24:06 +00:00
|
|
|
_vm->_isoMap->loadMulti(resourceData, resourceDataLength);
|
2005-01-28 19:25:41 +00:00
|
|
|
break;
|
2004-04-12 21:40:49 +00:00
|
|
|
case SAGA_PAL_ANIM:
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "Loading palette animation resource.");
|
2005-06-11 17:24:06 +00:00
|
|
|
_vm->_palanim->loadPalAnim(resourceData, resourceDataLength);
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
2004-10-05 02:16:26 +00:00
|
|
|
case SAGA_ENTRY:
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "Loading entry list resource...");
|
2005-06-11 17:24:06 +00:00
|
|
|
loadSceneEntryList(resourceData, resourceDataLength);
|
2004-10-05 02:16:26 +00:00
|
|
|
break;
|
|
|
|
case SAGA_FACES:
|
2005-07-19 19:05:52 +00:00
|
|
|
_vm->_interface->loadScenePortraits(_resourceList[i].resourceId);
|
2004-10-05 02:16:26 +00:00
|
|
|
break;
|
2005-07-05 15:15:35 +00:00
|
|
|
case SAGA_PALETTE:
|
|
|
|
{
|
2005-07-09 16:23:45 +00:00
|
|
|
PalEntry pal[PAL_ENTRIES];
|
2005-07-05 15:15:35 +00:00
|
|
|
byte *palPtr = resourceData;
|
|
|
|
|
|
|
|
if (resourceDataLength < 3 * PAL_ENTRIES)
|
2005-07-19 19:05:52 +00:00
|
|
|
error("Too small scene palette %i", resourceDataLength);
|
2005-07-05 15:15:35 +00:00
|
|
|
|
|
|
|
for (uint16 c = 0; c < PAL_ENTRIES; c++) {
|
|
|
|
pal[c].red = *palPtr++;
|
|
|
|
pal[c].green = *palPtr++;
|
|
|
|
pal[c].blue = *palPtr++;
|
|
|
|
}
|
|
|
|
_vm->_gfx->setPalette(pal);
|
|
|
|
}
|
|
|
|
break;
|
2004-04-12 21:40:49 +00:00
|
|
|
default:
|
2005-07-19 19:05:52 +00:00
|
|
|
error("Scene::ProcessSceneResources() Encountered unknown resource type %i", _resourceList[i].reourceType);
|
2004-04-12 21:40:49 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-05-22 11:59:22 +00:00
|
|
|
void Scene::draw() {
|
2005-07-09 16:23:45 +00:00
|
|
|
Surface *backBuffer;
|
|
|
|
Surface *backGroundSurface;
|
|
|
|
Rect rect;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-05-22 11:59:22 +00:00
|
|
|
backBuffer = _vm->_gfx->getBackBuffer();
|
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
backGroundSurface = _vm->_render->getBackGroundSurface();
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
if (_sceneDescription.flags & kSceneFlagISO) {
|
2005-02-18 00:00:00 +00:00
|
|
|
_vm->_isoMap->adjustScroll(false);
|
2005-05-22 11:59:22 +00:00
|
|
|
_vm->_isoMap->draw(backBuffer);
|
2005-01-09 23:41:22 +00:00
|
|
|
} else {
|
2005-07-09 16:23:45 +00:00
|
|
|
backGroundSurface->getRect(rect);
|
|
|
|
if (_sceneClip.bottom < rect.bottom) {
|
|
|
|
rect.bottom = _vm->getSceneHeight();
|
|
|
|
}
|
|
|
|
backBuffer->blit(rect, (const byte *)backGroundSurface->pixels);
|
2005-01-09 23:41:22 +00:00
|
|
|
}
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
void Scene::endScene() {
|
2005-07-09 16:23:45 +00:00
|
|
|
Surface *backBuffer;
|
|
|
|
Surface *backGroundSurface;
|
|
|
|
Rect rect;
|
2005-07-19 19:05:52 +00:00
|
|
|
size_t i;
|
2005-06-11 17:24:06 +00:00
|
|
|
|
2005-06-02 22:14:57 +00:00
|
|
|
if (!_sceneLoaded)
|
|
|
|
return;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-05 16:58:36 +00:00
|
|
|
debug(3, "Ending scene...");
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-03-29 22:14:20 +00:00
|
|
|
if (_sceneProc != NULL) {
|
2005-05-08 14:04:06 +00:00
|
|
|
_sceneProc(SCENE_END, this);
|
2005-03-29 22:14:20 +00:00
|
|
|
}
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-03-29 17:54:53 +00:00
|
|
|
//
|
|
|
|
_vm->_script->abortAllThreads();
|
|
|
|
_vm->_script->_skipSpeeches = false;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-04-22 01:38:27 +00:00
|
|
|
// Copy current screen to render buffer so inset rooms will get proper background
|
2005-07-09 16:23:45 +00:00
|
|
|
backBuffer= _vm->_gfx->getBackBuffer();
|
|
|
|
backGroundSurface = _vm->_render->getBackGroundSurface();
|
|
|
|
backBuffer->getRect(rect);
|
2005-04-22 01:38:27 +00:00
|
|
|
|
2005-07-09 16:23:45 +00:00
|
|
|
backGroundSurface->blit(rect, (const byte *)backBuffer->pixels);
|
2004-05-01 13:48:01 +00:00
|
|
|
// Free scene background
|
2004-08-04 20:28:57 +00:00
|
|
|
if (_bg.loaded) {
|
|
|
|
free(_bg.buf);
|
|
|
|
_bg.loaded = 0;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-05-01 13:48:01 +00:00
|
|
|
// Free scene background mask
|
2004-08-04 20:28:57 +00:00
|
|
|
if (_bgMask.loaded) {
|
|
|
|
free(_bgMask.buf);
|
|
|
|
_bgMask.loaded = 0;
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-05-01 13:48:01 +00:00
|
|
|
// Free scene resource list
|
2005-07-19 19:05:52 +00:00
|
|
|
for (i = 0; i < _resourceListCount; i++) {
|
|
|
|
free(_resourceList[i].buffer);
|
2005-06-11 17:24:06 +00:00
|
|
|
}
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-06-11 17:24:06 +00:00
|
|
|
if (_loadDescription) {
|
2005-07-19 19:05:52 +00:00
|
|
|
free(_resourceList);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-05-01 13:48:01 +00:00
|
|
|
// Free animation info list
|
2004-07-31 13:34:43 +00:00
|
|
|
_vm->_anim->reset();
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-08-10 19:20:33 +00:00
|
|
|
_vm->_palanim->freePalAnim();
|
2005-01-18 11:55:31 +00:00
|
|
|
|
|
|
|
_objectMap->freeMem();
|
2004-12-24 20:44:39 +00:00
|
|
|
_actionMap->freeMem();
|
2005-01-18 11:55:31 +00:00
|
|
|
_entryList.freeMem();
|
|
|
|
_sceneStrings.freeMem();
|
2005-03-18 17:11:37 +00:00
|
|
|
_vm->_isoMap->freeMem();
|
2004-10-09 07:39:46 +00:00
|
|
|
|
2004-08-10 19:00:30 +00:00
|
|
|
_vm->_events->clearList();
|
2005-07-14 17:46:21 +00:00
|
|
|
_textList.clear();
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2004-08-04 20:28:57 +00:00
|
|
|
_sceneLoaded = false;
|
2004-04-12 21:40:49 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2004-12-24 20:44:39 +00:00
|
|
|
void Scene::cmdSceneChange(int argc, const char **argv) {
|
2004-04-12 21:40:49 +00:00
|
|
|
int scene_num = 0;
|
|
|
|
|
2004-12-03 19:15:44 +00:00
|
|
|
scene_num = atoi(argv[1]);
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-07-08 18:39:53 +00:00
|
|
|
if ((scene_num < 1) || (scene_num >= _sceneCount)) {
|
2004-12-03 19:15:44 +00:00
|
|
|
_vm->_console->DebugPrintf("Invalid scene number.\n");
|
2004-04-12 21:40:49 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2004-08-04 20:28:57 +00:00
|
|
|
clearSceneQueue();
|
2004-04-12 21:40:49 +00:00
|
|
|
|
2005-05-08 14:04:06 +00:00
|
|
|
changeScene(scene_num, 0, kTransitionNoFade);
|
2004-04-12 21:40:49 +00:00
|
|
|
}
|
|
|
|
|
2004-12-24 20:44:39 +00:00
|
|
|
void Scene::cmdActionMapInfo() {
|
|
|
|
_actionMap->cmdInfo();
|
2004-10-07 22:31:39 +00:00
|
|
|
}
|
|
|
|
|
2004-12-24 20:44:39 +00:00
|
|
|
void Scene::cmdObjectMapInfo() {
|
|
|
|
_objectMap->cmdInfo();
|
2004-10-07 23:02:19 +00:00
|
|
|
}
|
|
|
|
|
2004-08-01 09:15:12 +00:00
|
|
|
|
2005-01-18 11:55:31 +00:00
|
|
|
void Scene::loadSceneEntryList(const byte* resourcePointer, size_t resourceLength) {
|
2005-01-16 19:06:04 +00:00
|
|
|
int i;
|
2005-01-18 11:55:31 +00:00
|
|
|
|
|
|
|
_entryList.entryListCount = resourceLength / 8;
|
2005-01-16 19:06:04 +00:00
|
|
|
|
2005-07-19 19:05:52 +00:00
|
|
|
MemoryReadStreamEndian readS(resourcePointer, resourceLength, _sceneContext->isBigEndian);
|
2005-01-16 19:06:04 +00:00
|
|
|
|
|
|
|
|
2005-01-18 11:55:31 +00:00
|
|
|
if (_entryList.entryList)
|
|
|
|
error("Scene::loadSceneEntryList entryList != NULL");
|
2005-01-16 19:06:04 +00:00
|
|
|
|
2005-01-18 11:55:31 +00:00
|
|
|
_entryList.entryList = (SceneEntry *) malloc(_entryList.entryListCount * sizeof(*_entryList.entryList));
|
|
|
|
if (_entryList.entryList == NULL) {
|
2005-01-21 21:55:54 +00:00
|
|
|
memoryError("Scene::loadSceneEntryList");
|
2005-01-16 19:06:04 +00:00
|
|
|
}
|
|
|
|
|
2005-01-18 11:55:31 +00:00
|
|
|
for (i = 0; i < _entryList.entryListCount; i++) {
|
|
|
|
_entryList.entryList[i].location.x = readS.readSint16();
|
|
|
|
_entryList.entryList[i].location.y = readS.readSint16();
|
|
|
|
_entryList.entryList[i].location.z = readS.readSint16();
|
|
|
|
_entryList.entryList[i].facing = readS.readUint16();
|
2005-01-16 19:06:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-04-12 21:40:49 +00:00
|
|
|
} // End of namespace Saga
|