2009-06-26 23:34:06 +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.
|
|
|
|
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* $URL$
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2009-06-27 01:04:24 +00:00
|
|
|
#include "common/stream.h"
|
|
|
|
|
|
|
|
#include "draci/draci.h"
|
2009-06-26 23:34:06 +00:00
|
|
|
#include "draci/game.h"
|
2009-06-27 01:04:24 +00:00
|
|
|
#include "draci/barchive.h"
|
2009-06-28 16:19:10 +00:00
|
|
|
#include "draci/script.h"
|
2009-07-03 17:54:13 +00:00
|
|
|
#include "draci/animation.h"
|
2009-06-27 01:04:24 +00:00
|
|
|
|
2009-07-19 13:28:05 +00:00
|
|
|
#include <cmath>
|
|
|
|
|
2009-06-27 01:04:24 +00:00
|
|
|
namespace Draci {
|
|
|
|
|
2009-09-30 10:45:14 +00:00
|
|
|
static const Common::String dialoguePath("ROZH");
|
2009-08-11 04:03:22 +00:00
|
|
|
|
2009-07-19 14:54:16 +00:00
|
|
|
static double real_to_double(byte real[6]);
|
|
|
|
|
2009-11-01 12:57:06 +00:00
|
|
|
enum {
|
|
|
|
kWalkingMapOverlayColour = 2,
|
|
|
|
kWalkingShortestPathOverlayColour = 120,
|
|
|
|
kWalkingObliquePathOverlayColour = 73
|
|
|
|
};
|
|
|
|
|
2009-06-28 16:19:10 +00:00
|
|
|
Game::Game(DraciEngine *vm) : _vm(vm) {
|
2009-09-30 10:45:14 +00:00
|
|
|
uint i;
|
|
|
|
|
2009-07-25 03:28:04 +00:00
|
|
|
BArchive *initArchive = _vm->_initArchive;
|
2009-09-28 03:51:23 +00:00
|
|
|
const BAFile *file;
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-06-27 05:10:24 +00:00
|
|
|
// Read in persons
|
2009-07-25 03:28:04 +00:00
|
|
|
file = initArchive->getFile(5);
|
2009-06-27 05:10:24 +00:00
|
|
|
Common::MemoryReadStream personData(file->_data, file->_length);
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-10-30 06:11:04 +00:00
|
|
|
const int personSize = sizeof(uint16) * 2 + sizeof(byte);
|
2009-09-30 10:45:14 +00:00
|
|
|
uint numPersons = file->_length / personSize;
|
2009-06-27 01:04:24 +00:00
|
|
|
_persons = new Person[numPersons];
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-06-27 05:10:24 +00:00
|
|
|
for (i = 0; i < numPersons; ++i) {
|
2009-08-02 05:21:21 +00:00
|
|
|
_persons[i]._x = personData.readUint16LE();
|
|
|
|
_persons[i]._y = personData.readUint16LE();
|
2009-08-04 15:41:27 +00:00
|
|
|
_persons[i]._fontColour = personData.readByte();
|
2009-06-27 05:10:24 +00:00
|
|
|
}
|
2009-07-17 01:20:51 +00:00
|
|
|
|
2009-08-11 04:03:22 +00:00
|
|
|
// Read in dialogue offsets
|
2009-07-25 03:28:04 +00:00
|
|
|
file = initArchive->getFile(4);
|
2009-08-11 04:03:22 +00:00
|
|
|
Common::MemoryReadStream dialogueData(file->_data, file->_length);
|
2009-09-30 10:45:14 +00:00
|
|
|
|
|
|
|
uint numDialogues = file->_length / sizeof(uint16);
|
2009-09-30 16:04:21 +00:00
|
|
|
_dialogueOffsets = new uint[numDialogues];
|
2009-09-30 10:45:14 +00:00
|
|
|
|
|
|
|
uint curOffset;
|
2009-08-11 04:03:22 +00:00
|
|
|
for (i = 0, curOffset = 0; i < numDialogues; ++i) {
|
|
|
|
_dialogueOffsets[i] = curOffset;
|
|
|
|
curOffset += dialogueData.readUint16LE();
|
2009-06-27 01:04:24 +00:00
|
|
|
}
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-08-11 04:03:22 +00:00
|
|
|
_dialogueVars = new int[curOffset];
|
|
|
|
memset(_dialogueVars, 0, sizeof (int) * curOffset);
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-06-27 05:10:24 +00:00
|
|
|
// Read in game info
|
2009-07-25 03:28:04 +00:00
|
|
|
file = initArchive->getFile(3);
|
2009-06-27 05:10:24 +00:00
|
|
|
Common::MemoryReadStream gameData(file->_data, file->_length);
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-07-14 00:51:35 +00:00
|
|
|
_info._startRoom = gameData.readByte() - 1;
|
2009-07-14 00:41:17 +00:00
|
|
|
_info._mapRoom = gameData.readByte() - 1;
|
|
|
|
_info._numObjects = gameData.readUint16LE();
|
2009-08-17 18:50:38 +00:00
|
|
|
_info._numItems = gameData.readUint16LE();
|
2009-07-14 00:41:17 +00:00
|
|
|
_info._numVariables = gameData.readByte();
|
|
|
|
_info._numPersons = gameData.readByte();
|
2009-08-11 04:03:22 +00:00
|
|
|
_info._numDialogues = gameData.readByte();
|
2009-08-17 18:50:38 +00:00
|
|
|
_info._maxItemWidth = gameData.readUint16LE();
|
|
|
|
_info._maxItemHeight = gameData.readUint16LE();
|
2009-07-14 00:41:17 +00:00
|
|
|
_info._musicLength = gameData.readUint16LE();
|
|
|
|
_info._crc[0] = gameData.readUint16LE();
|
|
|
|
_info._crc[1] = gameData.readUint16LE();
|
|
|
|
_info._crc[2] = gameData.readUint16LE();
|
|
|
|
_info._crc[3] = gameData.readUint16LE();
|
|
|
|
|
2009-08-11 04:03:22 +00:00
|
|
|
_info._numDialogueBlocks = curOffset;
|
2009-06-27 05:10:24 +00:00
|
|
|
|
|
|
|
// Read in variables
|
2009-07-25 03:28:04 +00:00
|
|
|
file = initArchive->getFile(2);
|
2009-09-30 10:45:14 +00:00
|
|
|
uint numVariables = file->_length / sizeof (int16);
|
2009-06-29 22:20:30 +00:00
|
|
|
|
2009-07-07 20:57:14 +00:00
|
|
|
_variables = new int[numVariables];
|
2009-06-29 22:20:30 +00:00
|
|
|
Common::MemoryReadStream variableData(file->_data, file->_length);
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-06-29 22:20:30 +00:00
|
|
|
for (i = 0; i < numVariables; ++i) {
|
|
|
|
_variables[i] = variableData.readUint16LE();
|
|
|
|
}
|
|
|
|
|
2009-07-13 19:24:22 +00:00
|
|
|
// Read in item icon status
|
2009-07-25 03:28:04 +00:00
|
|
|
file = initArchive->getFile(1);
|
2009-08-17 18:50:38 +00:00
|
|
|
uint numItems = file->_length;
|
2009-09-28 03:51:23 +00:00
|
|
|
_itemStatus = new byte[numItems];
|
|
|
|
memcpy(_itemStatus, file->_data, numItems);
|
2009-08-17 18:50:38 +00:00
|
|
|
_items = new GameItem[numItems];
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-06-27 05:10:24 +00:00
|
|
|
// Read in object status
|
2009-07-25 03:28:04 +00:00
|
|
|
file = initArchive->getFile(0);
|
2009-09-30 10:45:14 +00:00
|
|
|
uint numObjects = file->_length;
|
|
|
|
|
2009-07-02 15:08:42 +00:00
|
|
|
_objects = new GameObject[numObjects];
|
2009-09-30 16:04:21 +00:00
|
|
|
Common::MemoryReadStream objStatus(file->_data, file->_length);
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-07-02 15:08:42 +00:00
|
|
|
for (i = 0; i < numObjects; ++i) {
|
|
|
|
byte tmp = objStatus.readByte();
|
|
|
|
|
|
|
|
// Set object visibility
|
2009-07-06 19:50:59 +00:00
|
|
|
_objects[i]._visible = tmp & (1 << 7);
|
2009-07-02 15:08:42 +00:00
|
|
|
|
|
|
|
// Set object location
|
2009-07-06 19:50:59 +00:00
|
|
|
_objects[i]._location = (~(1 << 7) & tmp) - 1;
|
2009-09-30 10:45:14 +00:00
|
|
|
}
|
|
|
|
|
2009-08-11 04:03:22 +00:00
|
|
|
assert(numDialogues == _info._numDialogues);
|
2009-07-14 00:41:17 +00:00
|
|
|
assert(numPersons == _info._numPersons);
|
|
|
|
assert(numVariables == _info._numVariables);
|
|
|
|
assert(numObjects == _info._numObjects);
|
2009-09-30 10:45:14 +00:00
|
|
|
assert(numItems == _info._numItems);
|
2009-09-28 03:51:23 +00:00
|
|
|
|
|
|
|
// Deallocate all cached files, because we have copied them into our own data structures.
|
|
|
|
initArchive->clearCache();
|
2009-07-04 18:35:08 +00:00
|
|
|
}
|
2009-06-27 05:10:24 +00:00
|
|
|
|
2009-07-29 01:11:53 +00:00
|
|
|
void Game::start() {
|
|
|
|
while (!shouldQuit()) {
|
2009-10-01 08:32:35 +00:00
|
|
|
debugC(1, kDraciGeneralDebugLevel, "Game::start()");
|
|
|
|
|
2009-08-12 04:22:47 +00:00
|
|
|
// Whenever the top-level loop is entered, it should not finish unless
|
|
|
|
// the exit is triggered by a script
|
2009-10-28 00:04:59 +00:00
|
|
|
const bool force_reload = shouldExitLoop() > 1;
|
|
|
|
setExitLoop(false);
|
2009-10-26 04:59:52 +00:00
|
|
|
_vm->_script->endCurrentProgram(false);
|
2009-08-12 04:22:47 +00:00
|
|
|
|
2009-10-04 22:11:46 +00:00
|
|
|
enterNewRoom(force_reload);
|
2009-10-26 04:59:52 +00:00
|
|
|
|
|
|
|
if (_vm->_script->shouldEndProgram()) {
|
|
|
|
// Room changed during the initialization (intro or Escape pressed).
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2009-07-31 04:32:33 +00:00
|
|
|
loop();
|
2009-07-29 01:11:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-04 18:35:08 +00:00
|
|
|
void Game::init() {
|
2009-10-28 00:04:59 +00:00
|
|
|
setQuit(false);
|
|
|
|
setExitLoop(false);
|
2009-08-15 02:42:34 +00:00
|
|
|
_scheduledPalette = 0;
|
2009-10-28 07:34:17 +00:00
|
|
|
_fadePhases = _fadePhase = 0;
|
2009-10-29 15:26:48 +00:00
|
|
|
setEnableQuickHero(true);
|
|
|
|
setWantQuickHero(false);
|
|
|
|
setEnableSpeedText(true);
|
2009-10-01 16:47:34 +00:00
|
|
|
setLoopStatus(kStatusGate);
|
|
|
|
setLoopSubstatus(kSubstatusOrdinary);
|
2009-07-31 04:32:33 +00:00
|
|
|
|
2009-08-11 04:53:30 +00:00
|
|
|
_animUnderCursor = kOverlayImage;
|
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
_currentItem = kNoItem;
|
|
|
|
_itemUnderCursor = kNoItem;
|
2009-08-09 04:09:24 +00:00
|
|
|
|
2009-10-28 00:20:45 +00:00
|
|
|
_vm->_mouse->setCursorType(kHighlightedCursor); // anything different from kNormalCursor
|
2009-08-09 04:09:24 +00:00
|
|
|
|
2009-09-30 21:07:04 +00:00
|
|
|
_oldObjUnderCursor = _objUnderCursor = kOverlayImage;
|
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
// Set the inventory to empty initially
|
2009-09-30 10:45:14 +00:00
|
|
|
memset(_inventory, kNoItem, kInventorySlots * sizeof(int));
|
2009-08-17 18:50:38 +00:00
|
|
|
|
2009-07-29 01:11:53 +00:00
|
|
|
// Initialize animation for object / room titles
|
|
|
|
Animation *titleAnim = _vm->_anims->addText(kTitleText, true);
|
2009-10-30 01:41:57 +00:00
|
|
|
Text *title = new Text("", _vm->_smallFont, kTitleColour, 0, 0, 0);
|
2009-10-11 23:01:59 +00:00
|
|
|
titleAnim->addFrame(title, NULL);
|
2009-07-25 03:28:04 +00:00
|
|
|
|
2009-08-02 03:02:22 +00:00
|
|
|
// Initialize animation for speech text
|
|
|
|
Animation *speechAnim = _vm->_anims->addText(kSpeechText, true);
|
2009-10-30 01:41:57 +00:00
|
|
|
Text *speech = new Text("", _vm->_bigFont, kFontColour1, 0, 0, 0);
|
2009-10-11 23:01:59 +00:00
|
|
|
speechAnim->addFrame(speech, NULL);
|
2009-08-02 03:02:22 +00:00
|
|
|
|
2009-11-01 10:15:48 +00:00
|
|
|
// Initialize inventory animation. _iconsArchive is never flushed.
|
2009-09-28 03:51:23 +00:00
|
|
|
const BAFile *f = _vm->_iconsArchive->getFile(13);
|
2009-08-17 18:50:38 +00:00
|
|
|
Animation *inventoryAnim = _vm->_anims->addAnimation(kInventorySprite, 255, false);
|
|
|
|
Sprite *inventorySprite = new Sprite(f->_data, f->_length, 0, 0, true);
|
2009-10-11 23:01:59 +00:00
|
|
|
inventoryAnim->addFrame(inventorySprite, NULL);
|
2009-08-17 18:50:38 +00:00
|
|
|
inventoryAnim->setRelative((kScreenWidth - inventorySprite->getWidth()) / 2,
|
2009-09-30 10:45:14 +00:00
|
|
|
(kScreenHeight - inventorySprite->getHeight()) / 2);
|
2009-08-17 18:50:38 +00:00
|
|
|
|
2009-08-11 04:03:22 +00:00
|
|
|
for (uint i = 0; i < kDialogueLines; ++i) {
|
2009-08-17 18:50:38 +00:00
|
|
|
_dialogueAnims[i] = _vm->_anims->addText(kDialogueLinesID - i, true);
|
2009-10-30 01:41:57 +00:00
|
|
|
Text *dialogueLine = new Text("", _vm->_smallFont, kLineInactiveColour, 0, 0, 0);
|
2009-10-11 23:01:59 +00:00
|
|
|
_dialogueAnims[i]->addFrame(dialogueLine, NULL);
|
2009-08-11 04:03:22 +00:00
|
|
|
|
|
|
|
_dialogueAnims[i]->setZ(254);
|
2009-09-30 16:04:21 +00:00
|
|
|
_dialogueAnims[i]->setRelative(1,
|
2009-09-30 10:45:14 +00:00
|
|
|
kScreenHeight - (i + 1) * _vm->_smallFont->getFontHeight());
|
2009-08-11 04:03:22 +00:00
|
|
|
|
2009-10-30 01:56:52 +00:00
|
|
|
Text *text = reinterpret_cast<Text *>(_dialogueAnims[i]->getCurrentFrame());
|
2009-08-11 04:03:22 +00:00
|
|
|
text->setText("");
|
|
|
|
}
|
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
for (uint i = 0; i < _info._numItems; ++i) {
|
|
|
|
loadItem(i);
|
|
|
|
}
|
|
|
|
|
2009-07-07 19:50:12 +00:00
|
|
|
loadObject(kDragonObject);
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-09-28 04:09:03 +00:00
|
|
|
const GameObject *dragon = getObject(kDragonObject);
|
2009-07-15 19:06:24 +00:00
|
|
|
debugC(4, kDraciLogicDebugLevel, "Running init program for the dragon object...");
|
2009-07-07 19:53:40 +00:00
|
|
|
_vm->_script->run(dragon->_program, dragon->_init);
|
2009-07-04 23:05:13 +00:00
|
|
|
|
2009-10-01 08:32:35 +00:00
|
|
|
// Make sure we enter the right room in start().
|
2009-10-04 22:11:46 +00:00
|
|
|
setRoomNum(kNoEscRoom);
|
|
|
|
rememberRoomNumAsPrevious();
|
2009-10-01 08:32:35 +00:00
|
|
|
scheduleEnteringRoomUsingGate(_info._startRoom, 0);
|
2009-10-11 23:28:40 +00:00
|
|
|
_pushedNewRoom = _pushedNewGate = -1;
|
2009-06-27 01:04:24 +00:00
|
|
|
}
|
|
|
|
|
2009-07-17 00:20:57 +00:00
|
|
|
void Game::loop() {
|
2009-08-05 00:11:09 +00:00
|
|
|
Surface *surface = _vm->_screen->getSurface();
|
|
|
|
|
2009-07-29 19:38:02 +00:00
|
|
|
do {
|
2009-09-30 16:04:21 +00:00
|
|
|
debugC(4, kDraciLogicDebugLevel, "loopstatus: %d, loopsubstatus: %d",
|
2009-08-17 18:50:38 +00:00
|
|
|
_loopStatus, _loopSubstatus);
|
|
|
|
|
2009-07-29 19:38:02 +00:00
|
|
|
_vm->handleEvents();
|
2009-10-04 09:33:07 +00:00
|
|
|
if (shouldExitLoop() > 1) // after loading
|
2009-10-04 05:44:23 +00:00
|
|
|
break;
|
2009-07-29 01:11:53 +00:00
|
|
|
|
2009-10-28 07:34:17 +00:00
|
|
|
if (_fadePhase > 0 && (_vm->_system->getMillis() - _fadeTick) >= kFadingTimeUnit) {
|
|
|
|
_fadeTick = _vm->_system->getMillis();
|
|
|
|
--_fadePhase;
|
2009-10-29 01:55:06 +00:00
|
|
|
const byte *startPal = _currentRoom._palette >= 0 ? _vm->_paletteArchive->getFile(_currentRoom._palette)->_data : NULL;
|
|
|
|
const byte *endPal = getScheduledPalette() >= 0 ? _vm->_paletteArchive->getFile(getScheduledPalette())->_data : NULL;
|
|
|
|
_vm->_screen->interpolatePalettes(startPal, endPal, 0, kNumColours, _fadePhases - _fadePhase, _fadePhases);
|
2009-10-28 07:34:17 +00:00
|
|
|
if (_loopSubstatus == kSubstatusFade && _fadePhase == 0) {
|
|
|
|
setExitLoop(true);
|
|
|
|
// Rewrite the palette index of the current
|
|
|
|
// room. This is necessary when two fadings
|
|
|
|
// are called after each other, such as in the
|
2009-10-29 01:55:06 +00:00
|
|
|
// intro.
|
2009-10-28 07:34:17 +00:00
|
|
|
_currentRoom._palette = getScheduledPalette();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-30 10:45:14 +00:00
|
|
|
// Fetch mouse coordinates
|
2009-08-11 04:03:22 +00:00
|
|
|
int x = _vm->_mouse->getPosX();
|
|
|
|
int y = _vm->_mouse->getPosY();
|
2009-08-05 00:11:09 +00:00
|
|
|
|
2009-08-12 04:18:45 +00:00
|
|
|
if (_loopStatus == kStatusDialogue && _loopSubstatus == kSubstatusOrdinary) {
|
2009-08-11 04:53:30 +00:00
|
|
|
Text *text;
|
|
|
|
for (int i = 0; i < kDialogueLines; ++i) {
|
2009-10-30 01:56:52 +00:00
|
|
|
text = reinterpret_cast<Text *>(_dialogueAnims[i]->getCurrentFrame());
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-08-11 04:53:30 +00:00
|
|
|
if (_animUnderCursor == _dialogueAnims[i]->getID()) {
|
|
|
|
text->setColour(kLineActiveColour);
|
2009-08-12 05:38:15 +00:00
|
|
|
} else {
|
|
|
|
text->setColour(kLineInactiveColour);
|
2009-08-11 04:53:30 +00:00
|
|
|
}
|
|
|
|
}
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-08-11 04:03:22 +00:00
|
|
|
if (_vm->_mouse->lButtonPressed() || _vm->_mouse->rButtonPressed()) {
|
2009-10-28 00:04:59 +00:00
|
|
|
setExitLoop(true);
|
2009-08-11 04:03:22 +00:00
|
|
|
_vm->_mouse->lButtonSet(false);
|
|
|
|
_vm->_mouse->rButtonSet(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-30 16:16:53 +00:00
|
|
|
if (_vm->_mouse->isCursorOn()) {
|
2009-08-05 00:11:09 +00:00
|
|
|
// Fetch the dedicated objects' title animation / current frame
|
2009-07-29 19:38:02 +00:00
|
|
|
Animation *titleAnim = _vm->_anims->getAnimation(kTitleText);
|
2009-10-30 01:56:52 +00:00
|
|
|
Text *title = reinterpret_cast<Text *>(titleAnim->getCurrentFrame());
|
2009-07-27 04:51:34 +00:00
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
updateCursor();
|
|
|
|
updateTitle();
|
2009-08-06 03:52:08 +00:00
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
if (_loopStatus == kStatusOrdinary && _loopSubstatus == kSubstatusOrdinary) {
|
|
|
|
if (_vm->_mouse->lButtonPressed()) {
|
2009-09-30 10:45:14 +00:00
|
|
|
_vm->_mouse->lButtonSet(false);
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
if (_currentItem != kNoItem) {
|
|
|
|
putItem(_currentItem, 0);
|
|
|
|
_currentItem = kNoItem;
|
2009-08-09 04:09:24 +00:00
|
|
|
updateCursor();
|
2009-08-17 18:50:38 +00:00
|
|
|
} else {
|
2009-08-09 04:09:24 +00:00
|
|
|
if (_objUnderCursor != kObjectNotFound) {
|
2009-10-03 05:16:19 +00:00
|
|
|
const GameObject *obj = &_objects[_objUnderCursor];
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-08-09 04:09:24 +00:00
|
|
|
_vm->_mouse->cursorOff();
|
|
|
|
titleAnim->markDirtyRect(surface);
|
2009-08-06 04:50:37 +00:00
|
|
|
title->setText("");
|
2009-08-12 04:18:45 +00:00
|
|
|
_objUnderCursor = kObjectNotFound;
|
2009-08-06 04:50:37 +00:00
|
|
|
|
2009-08-08 12:31:49 +00:00
|
|
|
if (!obj->_imLook) {
|
2009-10-12 03:08:28 +00:00
|
|
|
if (obj->_lookDir == kDirectionLast) {
|
|
|
|
walkHero(x, y, obj->_lookDir);
|
2009-08-08 12:31:49 +00:00
|
|
|
} else {
|
2009-10-12 03:08:28 +00:00
|
|
|
walkHero(obj->_lookX, obj->_lookY, obj->_lookDir);
|
2009-08-08 12:31:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-06 03:52:08 +00:00
|
|
|
_vm->_script->run(obj->_program, obj->_look);
|
2009-08-06 04:50:37 +00:00
|
|
|
_vm->_mouse->cursorOn();
|
2009-08-09 04:09:24 +00:00
|
|
|
} else {
|
2009-10-12 03:08:28 +00:00
|
|
|
walkHero(x, y, kDirectionLast);
|
2009-08-06 03:52:08 +00:00
|
|
|
}
|
2009-08-09 04:09:24 +00:00
|
|
|
}
|
2009-08-17 18:50:38 +00:00
|
|
|
}
|
2009-08-06 03:52:08 +00:00
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
if (_vm->_mouse->rButtonPressed()) {
|
|
|
|
_vm->_mouse->rButtonSet(false);
|
2009-08-06 04:50:37 +00:00
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
if (_objUnderCursor != kObjectNotFound) {
|
2009-10-03 05:16:19 +00:00
|
|
|
const GameObject *obj = &_objects[_objUnderCursor];
|
2009-08-08 12:31:49 +00:00
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
if (_vm->_script->testExpression(obj->_program, obj->_canUse)) {
|
|
|
|
_vm->_mouse->cursorOff();
|
|
|
|
titleAnim->markDirtyRect(surface);
|
|
|
|
title->setText("");
|
|
|
|
_objUnderCursor = kObjectNotFound;
|
2009-08-08 12:31:49 +00:00
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
if (!obj->_imUse) {
|
2009-10-12 03:08:28 +00:00
|
|
|
if (obj->_useDir == kDirectionLast) {
|
|
|
|
walkHero(x, y, obj->_useDir);
|
2009-08-17 18:50:38 +00:00
|
|
|
} else {
|
2009-10-12 03:08:28 +00:00
|
|
|
walkHero(obj->_useX, obj->_useY, obj->_useDir);
|
2009-08-17 18:50:38 +00:00
|
|
|
}
|
2009-08-09 04:09:24 +00:00
|
|
|
}
|
2009-08-17 18:50:38 +00:00
|
|
|
|
|
|
|
_vm->_script->run(obj->_program, obj->_use);
|
|
|
|
_vm->_mouse->cursorOn();
|
2009-08-09 04:09:24 +00:00
|
|
|
} else {
|
2009-10-12 03:08:28 +00:00
|
|
|
walkHero(x, y, kDirectionLast);
|
2009-08-17 18:50:38 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (_vm->_script->testExpression(_currentRoom._program, _currentRoom._canUse)) {
|
|
|
|
_vm->_mouse->cursorOff();
|
|
|
|
titleAnim->markDirtyRect(surface);
|
|
|
|
title->setText("");
|
2009-08-06 05:04:21 +00:00
|
|
|
|
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
_vm->_script->run(_currentRoom._program, _currentRoom._use);
|
|
|
|
_vm->_mouse->cursorOn();
|
|
|
|
} else {
|
2009-10-12 03:08:28 +00:00
|
|
|
walkHero(x, y, kDirectionLast);
|
2009-08-09 04:09:24 +00:00
|
|
|
}
|
2009-08-06 05:04:21 +00:00
|
|
|
}
|
2009-07-29 19:38:02 +00:00
|
|
|
}
|
2009-07-27 05:34:22 +00:00
|
|
|
}
|
2009-07-27 04:51:34 +00:00
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
if (_loopStatus == kStatusInventory && _loopSubstatus == kSubstatusOrdinary) {
|
|
|
|
if (_inventoryExit) {
|
2009-09-30 10:45:14 +00:00
|
|
|
inventoryDone();
|
2009-08-17 18:50:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// If we are in inventory mode, all the animations except game items'
|
|
|
|
// images will necessarily be paused so we can safely assume that any
|
2009-09-30 16:04:21 +00:00
|
|
|
// animation under the cursor (a value returned by
|
2009-09-28 23:10:08 +00:00
|
|
|
// AnimationManager::getTopAnimationID()) will be an item animation or
|
2009-09-30 16:04:21 +00:00
|
|
|
// an overlay, for which we check. Item animations have their IDs
|
|
|
|
// calculated by offseting their itemID from the ID of the last "special"
|
|
|
|
// animation ID. In this way, we obtain its itemID.
|
2009-08-17 18:50:38 +00:00
|
|
|
if (_animUnderCursor != kOverlayImage && _animUnderCursor != kInventorySprite) {
|
|
|
|
_itemUnderCursor = kInventoryItemsID - _animUnderCursor;
|
|
|
|
} else {
|
2009-09-30 10:45:14 +00:00
|
|
|
_itemUnderCursor = kNoItem;
|
2009-08-17 18:50:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// If the user pressed the left mouse button
|
|
|
|
if (_vm->_mouse->lButtonPressed()) {
|
|
|
|
_vm->_mouse->lButtonSet(false);
|
|
|
|
|
|
|
|
// If there is an inventory item under the cursor and we aren't
|
|
|
|
// holding any item, run its look GPL program
|
|
|
|
if (_itemUnderCursor != kNoItem && _currentItem == kNoItem) {
|
2009-10-03 05:16:19 +00:00
|
|
|
const GameItem *item = &_items[_itemUnderCursor];
|
2009-08-17 18:50:38 +00:00
|
|
|
|
2009-10-03 05:16:19 +00:00
|
|
|
_vm->_script->run(item->_program, item->_look);
|
2009-08-17 18:50:38 +00:00
|
|
|
// Otherwise, if we are holding an item, try to place it inside the
|
|
|
|
// inventory
|
|
|
|
} else if (_currentItem != kNoItem) {
|
|
|
|
// FIXME: This should place the item in the nearest inventory slot,
|
|
|
|
// not the first one available
|
|
|
|
putItem(_currentItem, 0);
|
|
|
|
|
|
|
|
// Remove it from our hands
|
|
|
|
_currentItem = kNoItem;
|
|
|
|
}
|
|
|
|
} else if (_vm->_mouse->rButtonPressed()) {
|
|
|
|
_vm->_mouse->rButtonSet(false);
|
|
|
|
|
|
|
|
// If we right-clicked outside the inventory, close it
|
2009-09-28 23:10:08 +00:00
|
|
|
if (_animUnderCursor != kInventorySprite && _itemUnderCursor == kNoItem) {
|
2009-09-30 10:45:14 +00:00
|
|
|
inventoryDone();
|
2009-08-17 18:50:38 +00:00
|
|
|
|
|
|
|
// If there is an inventory item under our cursor
|
|
|
|
} else if (_itemUnderCursor != kNoItem) {
|
|
|
|
// Again, we have two possibilities:
|
|
|
|
|
|
|
|
// The first is that there is no item in our hands.
|
|
|
|
// In that case, just take the inventory item from the inventory.
|
|
|
|
if (_currentItem == kNoItem) {
|
|
|
|
_currentItem = _itemUnderCursor;
|
|
|
|
removeItem(_itemUnderCursor);
|
|
|
|
|
|
|
|
// The second is that there *is* an item in our hands.
|
|
|
|
// In that case, run the canUse script for the inventory item
|
|
|
|
// which will check if the two items are combinable and, finally,
|
|
|
|
// run the use script for the item.
|
|
|
|
} else {
|
2009-10-03 05:16:19 +00:00
|
|
|
const GameItem *item = &_items[_itemUnderCursor];
|
2009-08-17 18:50:38 +00:00
|
|
|
|
2009-10-03 05:16:19 +00:00
|
|
|
if (_vm->_script->testExpression(item->_program, item->_canUse)) {
|
|
|
|
_vm->_script->run(item->_program, item->_use);
|
2009-08-17 18:50:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
updateCursor();
|
|
|
|
}
|
|
|
|
}
|
2009-09-30 10:45:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-09-30 16:04:21 +00:00
|
|
|
debugC(5, kDraciLogicDebugLevel, "Anim under cursor: %d", _animUnderCursor);
|
2009-08-11 04:03:22 +00:00
|
|
|
|
2009-08-05 00:11:09 +00:00
|
|
|
// Handle character talking (if there is any)
|
2009-08-12 04:18:45 +00:00
|
|
|
if (_loopSubstatus == kSubstatusTalk) {
|
2009-09-30 16:04:21 +00:00
|
|
|
// If the current speech text has expired or the user clicked a mouse button,
|
2009-09-30 10:45:14 +00:00
|
|
|
// advance to the next line of text
|
2009-10-29 18:15:12 +00:00
|
|
|
if ((getEnableSpeedText() && (_vm->_mouse->lButtonPressed() || _vm->_mouse->rButtonPressed())) ||
|
2009-10-13 04:44:22 +00:00
|
|
|
(_vm->_system->getMillis() - _speechTick) >= _speechDuration) {
|
2009-08-06 05:17:18 +00:00
|
|
|
|
2009-10-28 00:04:59 +00:00
|
|
|
setExitLoop(true);
|
2009-08-02 05:21:21 +00:00
|
|
|
}
|
2009-10-29 15:26:48 +00:00
|
|
|
_vm->_mouse->lButtonSet(false);
|
|
|
|
_vm->_mouse->rButtonSet(false);
|
2009-08-02 05:21:21 +00:00
|
|
|
}
|
|
|
|
|
2009-08-05 00:11:09 +00:00
|
|
|
// This returns true if we got a signal to quit the game
|
2009-07-29 19:38:02 +00:00
|
|
|
if (shouldQuit())
|
|
|
|
return;
|
|
|
|
|
2009-08-05 00:11:09 +00:00
|
|
|
// Advance animations and redraw screen
|
2009-08-17 18:50:38 +00:00
|
|
|
_vm->_anims->drawScene(surface);
|
2009-07-29 19:38:02 +00:00
|
|
|
_vm->_screen->copyToScreen();
|
|
|
|
_vm->_system->delayMillis(20);
|
2009-07-29 01:11:53 +00:00
|
|
|
|
2009-07-31 04:32:33 +00:00
|
|
|
// HACK: Won't be needed once the game loop is implemented properly
|
2009-10-28 00:04:59 +00:00
|
|
|
setExitLoop(shouldExitLoop() || (_newRoom != getRoomNum() &&
|
|
|
|
(_loopStatus == kStatusOrdinary || _loopStatus == kStatusGate)));
|
2009-07-31 04:32:33 +00:00
|
|
|
|
2009-07-29 19:38:02 +00:00
|
|
|
} while (!shouldExitLoop());
|
2009-07-25 04:36:43 +00:00
|
|
|
}
|
2009-07-18 03:00:12 +00:00
|
|
|
|
2009-08-09 04:09:24 +00:00
|
|
|
void Game::updateCursor() {
|
2009-09-30 10:45:14 +00:00
|
|
|
// Fetch mouse coordinates
|
2009-08-17 18:50:38 +00:00
|
|
|
int x = _vm->_mouse->getPosX();
|
|
|
|
int y = _vm->_mouse->getPosY();
|
|
|
|
|
|
|
|
// Find animation under cursor
|
|
|
|
_animUnderCursor = _vm->_anims->getTopAnimationID(x, y);
|
|
|
|
|
|
|
|
// If we are inside a dialogue, all we need is to update the ID of the current
|
|
|
|
// animation under the cursor. This enables us to update the currently selected
|
|
|
|
// dialogue line (by recolouring it) but still leave the cursor unupdated when
|
|
|
|
// over background objects.
|
|
|
|
if (_loopStatus == kStatusDialogue)
|
|
|
|
return;
|
|
|
|
|
2009-10-27 23:51:32 +00:00
|
|
|
bool mouseChanged = false;
|
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
// If we are in inventory mode, we do a different kind of updating that handles
|
|
|
|
// inventory items and return early
|
|
|
|
if (_loopStatus == kStatusInventory && _loopSubstatus == kSubstatusOrdinary) {
|
|
|
|
if (_itemUnderCursor != kNoItem) {
|
2009-10-03 05:16:19 +00:00
|
|
|
const GameItem *item = &_items[_itemUnderCursor];
|
2009-08-17 18:50:38 +00:00
|
|
|
|
2009-10-03 05:16:19 +00:00
|
|
|
if (_vm->_script->testExpression(item->_program, item->_canUse)) {
|
2009-08-17 18:50:38 +00:00
|
|
|
if (_currentItem == kNoItem) {
|
2009-09-30 10:45:14 +00:00
|
|
|
_vm->_mouse->setCursorType(kHighlightedCursor);
|
2009-08-17 18:50:38 +00:00
|
|
|
} else {
|
2009-09-30 10:45:14 +00:00
|
|
|
_vm->_mouse->loadItemCursor(_currentItem, true);
|
2009-08-17 18:50:38 +00:00
|
|
|
}
|
2009-10-27 23:51:32 +00:00
|
|
|
mouseChanged = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!mouseChanged) {
|
|
|
|
if (_currentItem == kNoItem) {
|
|
|
|
_vm->_mouse->setCursorType(kNormalCursor);
|
|
|
|
} else {
|
2009-10-30 01:41:57 +00:00
|
|
|
_vm->_mouse->loadItemCursor(_currentItem, false);
|
2009-08-17 18:50:38 +00:00
|
|
|
}
|
|
|
|
}
|
2009-08-09 04:09:24 +00:00
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
return;
|
2009-08-09 04:09:24 +00:00
|
|
|
}
|
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
// Find the game object under the cursor
|
|
|
|
// (to be more precise, one that corresponds to the animation under the cursor)
|
|
|
|
int curObject = getObjectWithAnimation(_animUnderCursor);
|
|
|
|
|
|
|
|
// Update the game object under the cursor
|
|
|
|
_objUnderCursor = curObject;
|
|
|
|
if (_objUnderCursor != _oldObjUnderCursor) {
|
|
|
|
_oldObjUnderCursor = _objUnderCursor;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Handle main menu
|
|
|
|
|
|
|
|
// If there is no game object under the cursor, try using the room itself
|
2009-09-30 10:45:14 +00:00
|
|
|
if (_objUnderCursor == kObjectNotFound) {
|
2009-08-09 04:09:24 +00:00
|
|
|
if (_vm->_script->testExpression(_currentRoom._program, _currentRoom._canUse)) {
|
2009-08-17 18:50:38 +00:00
|
|
|
if (_currentItem == kNoItem) {
|
2009-08-09 04:09:24 +00:00
|
|
|
_vm->_mouse->setCursorType(kHighlightedCursor);
|
|
|
|
} else {
|
2009-08-17 18:50:38 +00:00
|
|
|
_vm->_mouse->loadItemCursor(_currentItem, true);
|
2009-08-09 04:09:24 +00:00
|
|
|
}
|
2009-10-27 23:51:32 +00:00
|
|
|
mouseChanged = true;
|
2009-08-09 04:09:24 +00:00
|
|
|
}
|
2009-08-17 18:50:38 +00:00
|
|
|
// If there *is* a game object under the cursor, update the cursor image
|
2009-08-09 04:09:24 +00:00
|
|
|
} else {
|
2009-10-03 05:16:19 +00:00
|
|
|
const GameObject *obj = &_objects[_objUnderCursor];
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
// If there is no walking direction set on the object (i.e. the object
|
|
|
|
// is not a gate / exit), test whether it can be used and, if so,
|
|
|
|
// update the cursor image (highlight it).
|
|
|
|
if (obj->_walkDir == 0) {
|
2009-08-09 04:09:24 +00:00
|
|
|
if (_vm->_script->testExpression(obj->_program, obj->_canUse)) {
|
2009-08-17 18:50:38 +00:00
|
|
|
if (_currentItem == kNoItem) {
|
2009-08-09 04:09:24 +00:00
|
|
|
_vm->_mouse->setCursorType(kHighlightedCursor);
|
|
|
|
} else {
|
2009-08-17 18:50:38 +00:00
|
|
|
_vm->_mouse->loadItemCursor(_currentItem, true);
|
2009-08-09 04:09:24 +00:00
|
|
|
}
|
2009-10-27 23:51:32 +00:00
|
|
|
mouseChanged = true;
|
2009-08-09 04:09:24 +00:00
|
|
|
}
|
2009-08-17 18:50:38 +00:00
|
|
|
// If the walking direction *is* set, the game object is a gate, so update
|
|
|
|
// the cursor image to the appropriate arrow.
|
|
|
|
} else {
|
|
|
|
_vm->_mouse->setCursorType((CursorType)obj->_walkDir);
|
2009-10-27 23:51:32 +00:00
|
|
|
mouseChanged = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Load the appropriate cursor (item image if an item is held or ordinary cursor
|
|
|
|
// if not)
|
|
|
|
if (!mouseChanged) {
|
|
|
|
if (_currentItem == kNoItem) {
|
|
|
|
_vm->_mouse->setCursorType(kNormalCursor);
|
|
|
|
} else {
|
2009-10-30 01:41:57 +00:00
|
|
|
_vm->_mouse->loadItemCursor(_currentItem, false);
|
2009-08-09 04:09:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Game::updateTitle() {
|
2009-08-17 18:50:38 +00:00
|
|
|
// If we are inside a dialogue, don't update titles
|
|
|
|
if (_loopStatus == kStatusDialogue)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Fetch current surface and height of the small font (used for titles)
|
2009-08-09 04:09:24 +00:00
|
|
|
Surface *surface = _vm->_screen->getSurface();
|
|
|
|
const int smallFontHeight = _vm->_smallFont->getFontHeight();
|
|
|
|
|
2009-09-30 10:45:14 +00:00
|
|
|
// Fetch mouse coordinates
|
2009-08-09 04:09:24 +00:00
|
|
|
int x = _vm->_mouse->getPosX();
|
|
|
|
int y = _vm->_mouse->getPosY();
|
|
|
|
|
|
|
|
// Fetch the dedicated objects' title animation / current frame
|
|
|
|
Animation *titleAnim = _vm->_anims->getAnimation(kTitleText);
|
2009-10-30 01:56:52 +00:00
|
|
|
Text *title = reinterpret_cast<Text *>(titleAnim->getCurrentFrame());
|
2009-08-09 04:09:24 +00:00
|
|
|
|
|
|
|
// Mark dirty rectangle to delete the previous text
|
2009-09-30 16:04:21 +00:00
|
|
|
titleAnim->markDirtyRect(surface);
|
|
|
|
|
2009-10-03 05:16:19 +00:00
|
|
|
if (_loopStatus == kStatusInventory) {
|
|
|
|
// If there is no item under the cursor, delete the title.
|
|
|
|
// Otherwise, show the item's title.
|
|
|
|
if (_itemUnderCursor == kNoItem) {
|
|
|
|
title->setText("");
|
|
|
|
} else {
|
|
|
|
const GameItem *item = &_items[_itemUnderCursor];
|
|
|
|
title->setText(item->_title);
|
|
|
|
}
|
2009-08-09 04:09:24 +00:00
|
|
|
} else {
|
2009-10-03 05:16:19 +00:00
|
|
|
// If there is no object under the cursor, delete the title.
|
|
|
|
// Otherwise, show the object's title.
|
|
|
|
if (_objUnderCursor == kObjectNotFound) {
|
|
|
|
title->setText("");
|
|
|
|
} else {
|
|
|
|
const GameObject *obj = &_objects[_objUnderCursor];
|
|
|
|
title->setText(obj->_title);
|
|
|
|
}
|
2009-08-09 04:09:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Move the title to the correct place (just above the cursor)
|
|
|
|
int newX = surface->centerOnX(x, title->getWidth());
|
2009-10-03 05:16:19 +00:00
|
|
|
int newY = surface->putAboveY(y - smallFontHeight / 2, title->getHeight());
|
2009-08-09 04:09:24 +00:00
|
|
|
titleAnim->setRelative(newX, newY);
|
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
// If we are currently playing the title, mark it dirty so it gets updated.
|
|
|
|
// Otherwise, start playing the title animation.
|
2009-08-09 04:09:24 +00:00
|
|
|
if (titleAnim->isPlaying()) {
|
|
|
|
titleAnim->markDirtyRect(surface);
|
|
|
|
} else {
|
|
|
|
_vm->_anims->play(titleAnim->getID());
|
|
|
|
}
|
2009-09-30 10:45:14 +00:00
|
|
|
}
|
2009-08-09 04:09:24 +00:00
|
|
|
|
2009-09-25 08:13:39 +00:00
|
|
|
int Game::getObjectWithAnimation(int animID) const {
|
2009-07-27 04:51:34 +00:00
|
|
|
for (uint i = 0; i < _info._numObjects; ++i) {
|
|
|
|
GameObject *obj = &_objects[i];
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-10-12 00:01:39 +00:00
|
|
|
for (uint j = 0; j < obj->_anim.size(); ++j) {
|
|
|
|
if (obj->_anim[j] == animID) {
|
2009-07-27 04:51:34 +00:00
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-05 17:58:14 +00:00
|
|
|
return kObjectNotFound;
|
2009-07-27 04:51:34 +00:00
|
|
|
}
|
2009-08-11 04:03:22 +00:00
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
void Game::removeItem(int itemID) {
|
|
|
|
for (uint i = 0; i < kInventorySlots; ++i) {
|
|
|
|
if (_inventory[i] == itemID) {
|
|
|
|
_inventory[i] = kNoItem;
|
|
|
|
_vm->_anims->stop(kInventoryItemsID - itemID);
|
2009-09-30 10:45:14 +00:00
|
|
|
break;
|
2009-08-17 18:50:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Game::putItem(int itemID, int position) {
|
|
|
|
if (itemID == kNoItem)
|
|
|
|
return;
|
|
|
|
|
2009-09-30 16:04:21 +00:00
|
|
|
if (position >= 0 &&
|
2009-08-17 18:50:38 +00:00
|
|
|
position < kInventoryLines * kInventoryColumns &&
|
2009-10-04 09:13:15 +00:00
|
|
|
(_inventory[position] == kNoItem || _inventory[position] == itemID)) {
|
2009-08-17 18:50:38 +00:00
|
|
|
_inventory[position] = itemID;
|
|
|
|
} else {
|
2009-10-04 09:13:15 +00:00
|
|
|
for (position = 0; position < kInventorySlots; ++position) {
|
|
|
|
if (_inventory[position] == kNoItem) {
|
|
|
|
_inventory[position] = itemID;
|
2009-09-30 10:45:14 +00:00
|
|
|
break;
|
2009-08-17 18:50:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-10-04 09:13:15 +00:00
|
|
|
const int line = position / kInventoryColumns + 1;
|
|
|
|
const int column = position % kInventoryColumns + 1;
|
2009-08-17 18:50:38 +00:00
|
|
|
|
2009-10-04 09:13:15 +00:00
|
|
|
const int anim_id = kInventoryItemsID - itemID;
|
|
|
|
Animation *anim = _vm->_anims->getAnimation(anim_id);
|
|
|
|
if (!anim) {
|
2009-10-30 01:41:57 +00:00
|
|
|
anim = _vm->_anims->addItem(anim_id, false);
|
2009-11-01 10:15:48 +00:00
|
|
|
// _itemImagesArchive is never flushed.
|
2009-10-04 09:13:15 +00:00
|
|
|
const BAFile *img = _vm->_itemImagesArchive->getFile(2 * itemID);
|
|
|
|
Sprite *sp = new Sprite(img->_data, img->_length, 0, 0, true);
|
2009-10-11 23:01:59 +00:00
|
|
|
anim->addFrame(sp, NULL);
|
2009-10-04 09:13:15 +00:00
|
|
|
}
|
2009-10-30 01:56:52 +00:00
|
|
|
Drawable *frame = anim->getCurrentFrame();
|
2009-08-17 18:50:38 +00:00
|
|
|
|
2009-09-30 16:04:21 +00:00
|
|
|
const int x = kInventoryX +
|
2009-09-30 10:45:14 +00:00
|
|
|
(column * kInventoryItemWidth) -
|
|
|
|
(kInventoryItemWidth / 2) -
|
|
|
|
(frame->getWidth() / 2);
|
2009-08-17 18:50:38 +00:00
|
|
|
|
2009-09-30 16:04:21 +00:00
|
|
|
const int y = kInventoryY +
|
|
|
|
(line * kInventoryItemHeight) -
|
2009-09-30 10:45:14 +00:00
|
|
|
(kInventoryItemHeight / 2) -
|
|
|
|
(frame->getHeight() / 2);
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
debug(2, "itemID: %d position: %d line: %d column: %d x: %d y: %d", itemID, position, line, column, x, y);
|
|
|
|
|
|
|
|
anim->setRelative(x, y);
|
|
|
|
|
|
|
|
// If we are in inventory mode, we need to play the item animation, immediately
|
|
|
|
// upon returning it to its slot but *not* in other modes because it should be
|
|
|
|
// invisible then (along with the inventory)
|
|
|
|
if (_loopStatus == kStatusInventory && _loopSubstatus == kSubstatusOrdinary) {
|
2009-10-04 09:13:15 +00:00
|
|
|
_vm->_anims->play(anim_id);
|
2009-08-17 18:50:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Game::inventoryInit() {
|
|
|
|
// Pause all "background" animations
|
|
|
|
_vm->_anims->pauseAnimations();
|
|
|
|
|
|
|
|
// Draw the inventory and the current items
|
|
|
|
inventoryDraw();
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
// Turn cursor on if it is off
|
|
|
|
_vm->_mouse->cursorOn();
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
// Set the appropriate loop status
|
2009-10-01 08:32:35 +00:00
|
|
|
setLoopStatus(kStatusInventory);
|
2009-08-17 18:50:38 +00:00
|
|
|
|
|
|
|
// TODO: This will be used for exiting the inventory automatically when the mouse
|
|
|
|
// is outside it for some time
|
|
|
|
_inventoryExit = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Game::inventoryDone() {
|
|
|
|
_vm->_mouse->cursorOn();
|
2009-10-01 08:32:35 +00:00
|
|
|
setLoopStatus(kStatusOrdinary);
|
2009-08-17 18:50:38 +00:00
|
|
|
|
|
|
|
_vm->_anims->unpauseAnimations();
|
|
|
|
|
|
|
|
_vm->_anims->stop(kInventorySprite);
|
|
|
|
|
|
|
|
for (uint i = 0; i < kInventorySlots; ++i) {
|
|
|
|
if (_inventory[i] != kNoItem) {
|
|
|
|
_vm->_anims->stop(kInventoryItemsID - _inventory[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reset item under cursor
|
|
|
|
_itemUnderCursor = kNoItem;
|
|
|
|
|
|
|
|
// TODO: Handle main menu
|
|
|
|
}
|
|
|
|
|
|
|
|
void Game::inventoryDraw() {
|
|
|
|
_vm->_anims->play(kInventorySprite);
|
|
|
|
|
|
|
|
for (uint i = 0; i < kInventorySlots; ++i) {
|
|
|
|
if (_inventory[i] != kNoItem) {
|
|
|
|
_vm->_anims->play(kInventoryItemsID - _inventory[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-10-04 09:13:15 +00:00
|
|
|
void Game::inventoryReload() {
|
|
|
|
// Make sure all items are loaded into memory (e.g., after loading a
|
|
|
|
// savegame) by re-putting them on the same spot in the inventory.
|
|
|
|
for (uint i = 0; i < kInventorySlots; ++i) {
|
|
|
|
putItem(_inventory[i], i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-11 04:03:22 +00:00
|
|
|
void Game::dialogueMenu(int dialogueID) {
|
|
|
|
int oldLines, hit;
|
|
|
|
|
|
|
|
char tmp[5];
|
|
|
|
sprintf(tmp, "%d", dialogueID+1);
|
|
|
|
Common::String ext(tmp);
|
|
|
|
_dialogueArchive = new BArchive(dialoguePath + ext + ".dfw");
|
|
|
|
|
2009-09-30 16:04:21 +00:00
|
|
|
debugC(4, kDraciLogicDebugLevel, "Starting dialogue (ID: %d, Archive: %s)",
|
2009-09-30 10:45:14 +00:00
|
|
|
dialogueID, (dialoguePath + ext + ".dfw").c_str());
|
2009-08-11 04:03:22 +00:00
|
|
|
|
|
|
|
_currentDialogue = dialogueID;
|
|
|
|
oldLines = 255;
|
|
|
|
dialogueInit(dialogueID);
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-08-11 04:03:22 +00:00
|
|
|
do {
|
|
|
|
_dialogueExit = false;
|
|
|
|
hit = dialogueDraw();
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-09-30 16:04:21 +00:00
|
|
|
debugC(7, kDraciLogicDebugLevel,
|
|
|
|
"hit: %d, _lines[hit]: %d, lastblock: %d, dialogueLines: %d, dialogueExit: %d",
|
2009-08-17 19:37:55 +00:00
|
|
|
hit, _lines[hit], _lastBlock, _dialogueLinesNum, _dialogueExit);
|
2009-08-11 04:03:22 +00:00
|
|
|
|
|
|
|
if ((!_dialogueExit) && (hit != -1) && (_lines[hit] != -1)) {
|
2009-08-17 19:37:55 +00:00
|
|
|
if ((oldLines == 1) && (_dialogueLinesNum == 1) && (_lines[hit] == _lastBlock)) {
|
2009-08-11 04:03:22 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
_currentBlock = _lines[hit];
|
|
|
|
runDialogueProg(_dialogueBlocks[_lines[hit]]._program, 1);
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
_lastBlock = _lines[hit];
|
2009-08-12 07:34:31 +00:00
|
|
|
_dialogueVars[_dialogueOffsets[dialogueID] + _lastBlock] += 1;
|
2009-08-11 04:03:22 +00:00
|
|
|
_dialogueBegin = false;
|
2009-08-17 19:37:55 +00:00
|
|
|
oldLines = _dialogueLinesNum;
|
2009-08-11 04:03:22 +00:00
|
|
|
|
2009-09-30 16:16:53 +00:00
|
|
|
} while (!_dialogueExit);
|
2009-08-11 04:03:22 +00:00
|
|
|
|
|
|
|
dialogueDone();
|
|
|
|
_currentDialogue = kNoDialogue;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Game::dialogueDraw() {
|
2009-08-17 19:37:55 +00:00
|
|
|
_dialogueLinesNum = 0;
|
2009-08-11 04:03:22 +00:00
|
|
|
int i = 0;
|
|
|
|
int ret = 0;
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-08-11 04:03:22 +00:00
|
|
|
Animation *anim;
|
|
|
|
Text *dialogueLine;
|
|
|
|
|
2009-08-17 19:37:55 +00:00
|
|
|
while ((_dialogueLinesNum < 4) && (i < _blockNum)) {
|
2009-08-11 04:03:22 +00:00
|
|
|
GPL2Program blockTest;
|
|
|
|
blockTest._bytecode = _dialogueBlocks[i]._canBlock;
|
|
|
|
blockTest._length = _dialogueBlocks[i]._canLen;
|
|
|
|
debugC(3, kDraciLogicDebugLevel, "Testing dialogue block %d", i);
|
|
|
|
if (_vm->_script->testExpression(blockTest, 1)) {
|
2009-08-17 19:37:55 +00:00
|
|
|
anim = _dialogueAnims[_dialogueLinesNum];
|
2009-10-30 01:56:52 +00:00
|
|
|
dialogueLine = reinterpret_cast<Text *>(anim->getCurrentFrame());
|
2009-08-11 04:03:22 +00:00
|
|
|
dialogueLine->setText(_dialogueBlocks[i]._title);
|
|
|
|
|
|
|
|
dialogueLine->setColour(kLineInactiveColour);
|
2009-08-17 19:37:55 +00:00
|
|
|
_lines[_dialogueLinesNum] = i;
|
|
|
|
_dialogueLinesNum++;
|
2009-08-11 04:03:22 +00:00
|
|
|
}
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
|
2009-08-17 19:37:55 +00:00
|
|
|
for (i = _dialogueLinesNum; i < kDialogueLines; ++i) {
|
2009-08-12 07:34:31 +00:00
|
|
|
_lines[i] = -1;
|
2009-08-11 04:03:22 +00:00
|
|
|
anim = _dialogueAnims[i];
|
2009-10-30 01:56:52 +00:00
|
|
|
dialogueLine = reinterpret_cast<Text *>(anim->getCurrentFrame());
|
2009-08-11 04:03:22 +00:00
|
|
|
dialogueLine->setText("");
|
|
|
|
}
|
|
|
|
|
|
|
|
_oldObjUnderCursor = kObjectNotFound;
|
|
|
|
|
2009-08-17 19:37:55 +00:00
|
|
|
if (_dialogueLinesNum > 1) {
|
2009-08-11 04:03:22 +00:00
|
|
|
_vm->_mouse->cursorOn();
|
2009-10-28 00:04:59 +00:00
|
|
|
setExitLoop(false);
|
2009-08-11 04:03:22 +00:00
|
|
|
loop();
|
|
|
|
_vm->_mouse->cursorOff();
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-08-11 04:03:22 +00:00
|
|
|
bool notDialogueAnim = true;
|
|
|
|
for (uint j = 0; j < kDialogueLines; ++j) {
|
|
|
|
if (_dialogueAnims[j]->getID() == _animUnderCursor) {
|
|
|
|
notDialogueAnim = false;
|
|
|
|
break;
|
|
|
|
}
|
2009-09-30 10:45:14 +00:00
|
|
|
}
|
|
|
|
|
2009-08-11 04:03:22 +00:00
|
|
|
if (notDialogueAnim) {
|
|
|
|
ret = -1;
|
|
|
|
} else {
|
|
|
|
ret = _dialogueAnims[0]->getID() - _animUnderCursor;
|
|
|
|
}
|
|
|
|
} else {
|
2009-08-17 19:37:55 +00:00
|
|
|
ret = _dialogueLinesNum - 1;
|
2009-08-11 04:03:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < kDialogueLines; ++i) {
|
2009-10-30 01:56:52 +00:00
|
|
|
dialogueLine = reinterpret_cast<Text *>(_dialogueAnims[i]->getCurrentFrame());
|
2009-08-11 04:03:22 +00:00
|
|
|
_dialogueAnims[i]->markDirtyRect(_vm->_screen->getSurface());
|
|
|
|
dialogueLine->setText("");
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Game::dialogueInit(int dialogID) {
|
|
|
|
_vm->_mouse->setCursorType(kDialogueCursor);
|
|
|
|
|
|
|
|
_blockNum = _dialogueArchive->size() / 3;
|
|
|
|
_dialogueBlocks = new Dialogue[_blockNum];
|
|
|
|
|
2009-09-28 03:51:23 +00:00
|
|
|
const BAFile *f;
|
2009-08-11 04:03:22 +00:00
|
|
|
|
|
|
|
for (uint i = 0; i < kDialogueLines; ++i) {
|
|
|
|
_lines[i] = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < _blockNum; ++i) {
|
|
|
|
f = _dialogueArchive->getFile(i * 3);
|
|
|
|
_dialogueBlocks[i]._canLen = f->_length;
|
|
|
|
_dialogueBlocks[i]._canBlock = f->_data;
|
|
|
|
|
|
|
|
f = _dialogueArchive->getFile(i * 3 + 1);
|
|
|
|
|
|
|
|
// The first byte of the file is the length of the string (without the length)
|
|
|
|
assert(f->_length - 1 == f->_data[0]);
|
|
|
|
|
|
|
|
_dialogueBlocks[i]._title = Common::String((char *)(f->_data+1), f->_length-1);
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-08-11 04:03:22 +00:00
|
|
|
f = _dialogueArchive->getFile(i * 3 + 2);
|
|
|
|
_dialogueBlocks[i]._program._bytecode = f->_data;
|
|
|
|
_dialogueBlocks[i]._program._length = f->_length;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint i = 0; i < kDialogueLines; ++i) {
|
|
|
|
_vm->_anims->play(_dialogueAnims[i]->getID());
|
|
|
|
}
|
|
|
|
|
2009-10-01 08:32:35 +00:00
|
|
|
setLoopStatus(kStatusDialogue);
|
2009-08-12 07:34:31 +00:00
|
|
|
_lastBlock = -1;
|
2009-08-11 04:03:22 +00:00
|
|
|
_dialogueBegin = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Game::dialogueDone() {
|
|
|
|
for (uint i = 0; i < kDialogueLines; ++i) {
|
|
|
|
_vm->_anims->stop(_dialogueAnims[i]->getID());
|
|
|
|
}
|
|
|
|
|
2009-10-01 16:47:34 +00:00
|
|
|
delete _dialogueArchive;
|
2009-08-11 04:03:22 +00:00
|
|
|
delete[] _dialogueBlocks;
|
|
|
|
|
2009-10-01 08:32:35 +00:00
|
|
|
setLoopStatus(kStatusOrdinary);
|
2009-08-11 04:03:22 +00:00
|
|
|
_vm->_mouse->setCursorType(kNormalCursor);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Game::runDialogueProg(GPL2Program prog, int offset) {
|
|
|
|
// Mark last animation
|
|
|
|
int lastAnimIndex = _vm->_anims->getLastIndex();
|
|
|
|
|
2009-08-12 05:22:50 +00:00
|
|
|
// Run the dialogue program
|
2009-08-11 04:03:22 +00:00
|
|
|
_vm->_script->run(prog, offset);
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-09-27 21:25:34 +00:00
|
|
|
deleteAnimationsAfterIndex(lastAnimIndex);
|
2009-08-11 04:03:22 +00:00
|
|
|
}
|
|
|
|
|
2009-10-12 03:08:28 +00:00
|
|
|
void Game::playHeroAnimation(int anim_index) {
|
|
|
|
const GameObject *dragon = getObject(kDragonObject);
|
|
|
|
const int animID = dragon->_anim[anim_index];
|
|
|
|
Animation *anim = _vm->_anims->getAnimation(animID);
|
|
|
|
stopObjectAnimations(dragon);
|
|
|
|
positionAnimAsHero(anim);
|
|
|
|
_vm->_anims->play(animID);
|
|
|
|
}
|
|
|
|
|
2009-11-01 12:57:06 +00:00
|
|
|
void Game::redrawWalkingPath(int id, byte colour, const WalkingMap::Path &path) {
|
|
|
|
Animation *anim = _vm->_anims->getAnimation(id);
|
|
|
|
Sprite *ov = _walkingMap.newOverlayFromPath(path, colour);
|
|
|
|
delete anim->getFrame(0);
|
|
|
|
anim->replaceFrame(0, ov, NULL);
|
|
|
|
anim->markDirtyRect(_vm->_screen->getSurface());
|
|
|
|
}
|
|
|
|
|
2009-10-12 03:08:28 +00:00
|
|
|
void Game::walkHero(int x, int y, SightDirection dir) {
|
2009-09-30 04:33:52 +00:00
|
|
|
// Needed for the map room with empty walking map. For some reason,
|
|
|
|
// findNearestWalkable() takes several seconds with 100% CPU to finish
|
|
|
|
// (correctly).
|
|
|
|
if (!_currentRoom._heroOn)
|
|
|
|
return;
|
2009-08-08 12:31:49 +00:00
|
|
|
|
2009-11-01 12:57:06 +00:00
|
|
|
Common::Point oldHero = _hero;
|
2009-08-08 12:31:49 +00:00
|
|
|
Surface *surface = _vm->_screen->getSurface();
|
2009-10-30 02:15:41 +00:00
|
|
|
_hero = _walkingMap.findNearestWalkable(x, y, surface->getDimensions());
|
2009-09-29 05:54:59 +00:00
|
|
|
debugC(3, kDraciLogicDebugLevel, "Walk to x: %d y: %d", _hero.x, _hero.y);
|
2009-07-25 04:36:43 +00:00
|
|
|
// FIXME: Need to add proper walking (this only warps the dragon to position)
|
2009-07-20 17:25:57 +00:00
|
|
|
|
2009-11-01 12:57:06 +00:00
|
|
|
// Compute the shortest and obliqued path.
|
|
|
|
WalkingMap::Path shortestPath, obliquePath;
|
2009-11-01 19:22:41 +00:00
|
|
|
_walkingMap.findShortestPath(oldHero, _hero, &shortestPath);
|
2009-11-01 21:19:39 +00:00
|
|
|
// TODO: test reachability and react
|
2009-11-01 12:57:06 +00:00
|
|
|
_walkingMap.obliquePath(shortestPath, &obliquePath);
|
2009-11-01 19:22:41 +00:00
|
|
|
if (_vm->_showWalkingMap) {
|
|
|
|
redrawWalkingPath(kWalkingShortestPathOverlay, kWalkingShortestPathOverlayColour, shortestPath);
|
|
|
|
redrawWalkingPath(kWalkingObliquePathOverlay, kWalkingObliquePathOverlayColour, obliquePath);
|
|
|
|
}
|
2009-11-01 12:57:06 +00:00
|
|
|
|
2009-10-12 03:08:28 +00:00
|
|
|
Movement movement = kStopRight;
|
|
|
|
switch (dir) {
|
|
|
|
case kDirectionLeft:
|
|
|
|
movement = kStopLeft;
|
|
|
|
break;
|
|
|
|
case kDirectionRight:
|
|
|
|
movement = kStopRight;
|
|
|
|
break;
|
|
|
|
default: {
|
|
|
|
const GameObject *dragon = getObject(kDragonObject);
|
|
|
|
const int anim_index = playingObjectAnimation(dragon);
|
|
|
|
if (anim_index >= 0) {
|
|
|
|
movement = static_cast<Movement> (anim_index);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
playHeroAnimation(movement);
|
2009-07-17 00:20:57 +00:00
|
|
|
}
|
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
void Game::loadItem(int itemID) {
|
2009-09-28 03:51:23 +00:00
|
|
|
const BAFile *f = _vm->_itemsArchive->getFile(itemID * 3);
|
2009-08-17 18:50:38 +00:00
|
|
|
Common::MemoryReadStream itemReader(f->_data, f->_length);
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-08-17 18:50:38 +00:00
|
|
|
GameItem *item = _items + itemID;
|
|
|
|
|
|
|
|
item->_init = itemReader.readSint16LE();
|
|
|
|
item->_look = itemReader.readSint16LE();
|
|
|
|
item->_use = itemReader.readSint16LE();
|
|
|
|
item->_canUse = itemReader.readSint16LE();
|
|
|
|
item->_imInit = itemReader.readByte();
|
|
|
|
item->_imLook = itemReader.readByte();
|
|
|
|
item->_imUse = itemReader.readByte();
|
|
|
|
|
|
|
|
f = _vm->_itemsArchive->getFile(itemID * 3 + 1);
|
|
|
|
|
|
|
|
// The first byte is the length of the string
|
|
|
|
item->_title = Common::String((const char *)f->_data + 1, f->_length - 1);
|
|
|
|
assert(f->_data[0] == item->_title.size());
|
|
|
|
|
|
|
|
f = _vm->_itemsArchive->getFile(itemID * 3 + 2);
|
|
|
|
|
|
|
|
item->_program._bytecode = f->_data;
|
|
|
|
item->_program._length = f->_length;
|
|
|
|
}
|
|
|
|
|
2009-07-13 19:53:53 +00:00
|
|
|
void Game::loadRoom(int roomNum) {
|
2009-09-28 03:51:23 +00:00
|
|
|
const BAFile *f;
|
2009-07-04 15:21:12 +00:00
|
|
|
f = _vm->_roomsArchive->getFile(roomNum * 4);
|
|
|
|
Common::MemoryReadStream roomReader(f->_data, f->_length);
|
|
|
|
|
|
|
|
roomReader.readUint32LE(); // Pointer to room program, not used
|
|
|
|
roomReader.readUint16LE(); // Program length, not used
|
|
|
|
roomReader.readUint32LE(); // Pointer to room title, not used
|
|
|
|
|
2009-10-22 07:34:43 +00:00
|
|
|
// Music will be played by the GPL2 command startMusic when needed.
|
|
|
|
setMusicTrack(roomReader.readByte());
|
2009-07-16 18:31:15 +00:00
|
|
|
|
2009-10-30 00:52:05 +00:00
|
|
|
_currentRoom._mapID = roomReader.readByte() - 1;
|
2009-07-04 15:21:12 +00:00
|
|
|
_currentRoom._palette = roomReader.readByte() - 1;
|
2009-07-17 00:27:21 +00:00
|
|
|
_currentRoom._numOverlays = roomReader.readSint16LE();
|
2009-07-15 19:06:24 +00:00
|
|
|
_currentRoom._init = roomReader.readSint16LE();
|
|
|
|
_currentRoom._look = roomReader.readSint16LE();
|
|
|
|
_currentRoom._use = roomReader.readSint16LE();
|
|
|
|
_currentRoom._canUse = roomReader.readSint16LE();
|
2009-07-04 15:21:12 +00:00
|
|
|
_currentRoom._imInit = roomReader.readByte();
|
|
|
|
_currentRoom._imLook = roomReader.readByte();
|
|
|
|
_currentRoom._imUse = roomReader.readByte();
|
|
|
|
_currentRoom._mouseOn = roomReader.readByte();
|
|
|
|
_currentRoom._heroOn = roomReader.readByte();
|
2009-07-19 13:28:05 +00:00
|
|
|
|
|
|
|
// Read in pers0 and persStep (stored as 6-byte Pascal reals)
|
|
|
|
byte real[6];
|
|
|
|
|
|
|
|
for (int i = 5; i >= 0; --i) {
|
|
|
|
real[i] = roomReader.readByte();
|
|
|
|
}
|
|
|
|
|
|
|
|
_currentRoom._pers0 = real_to_double(real);
|
|
|
|
|
|
|
|
for (int i = 5; i >= 0; --i) {
|
|
|
|
real[i] = roomReader.readByte();
|
|
|
|
}
|
|
|
|
|
|
|
|
_currentRoom._persStep = real_to_double(real);
|
|
|
|
|
2009-07-04 15:21:12 +00:00
|
|
|
_currentRoom._escRoom = roomReader.readByte() - 1;
|
|
|
|
_currentRoom._numGates = roomReader.readByte();
|
|
|
|
|
2009-10-22 07:34:43 +00:00
|
|
|
debugC(4, kDraciLogicDebugLevel, "Music: %d", getMusicTrack());
|
2009-10-30 02:15:41 +00:00
|
|
|
debugC(4, kDraciLogicDebugLevel, "Map: %d", getMapID());
|
2009-07-16 18:31:15 +00:00
|
|
|
debugC(4, kDraciLogicDebugLevel, "Palette: %d", _currentRoom._palette);
|
2009-07-17 00:27:21 +00:00
|
|
|
debugC(4, kDraciLogicDebugLevel, "Overlays: %d", _currentRoom._numOverlays);
|
2009-07-16 18:31:15 +00:00
|
|
|
debugC(4, kDraciLogicDebugLevel, "Init: %d", _currentRoom._init);
|
|
|
|
debugC(4, kDraciLogicDebugLevel, "Look: %d", _currentRoom._look);
|
|
|
|
debugC(4, kDraciLogicDebugLevel, "Use: %d", _currentRoom._use);
|
|
|
|
debugC(4, kDraciLogicDebugLevel, "CanUse: %d", _currentRoom._canUse);
|
|
|
|
debugC(4, kDraciLogicDebugLevel, "ImInit: %d", _currentRoom._imInit);
|
|
|
|
debugC(4, kDraciLogicDebugLevel, "ImLook: %d", _currentRoom._imLook);
|
|
|
|
debugC(4, kDraciLogicDebugLevel, "ImUse: %d", _currentRoom._imUse);
|
|
|
|
debugC(4, kDraciLogicDebugLevel, "MouseOn: %d", _currentRoom._mouseOn);
|
|
|
|
debugC(4, kDraciLogicDebugLevel, "HeroOn: %d", _currentRoom._heroOn);
|
|
|
|
debugC(4, kDraciLogicDebugLevel, "Pers0: %f", _currentRoom._pers0);
|
|
|
|
debugC(4, kDraciLogicDebugLevel, "PersStep: %f", _currentRoom._persStep);
|
|
|
|
debugC(4, kDraciLogicDebugLevel, "EscRoom: %d", _currentRoom._escRoom);
|
|
|
|
debugC(4, kDraciLogicDebugLevel, "Gates: %d", _currentRoom._numGates);
|
2009-07-15 19:06:24 +00:00
|
|
|
|
2009-07-18 03:05:20 +00:00
|
|
|
// Read in the gates' numbers
|
2009-07-31 04:32:33 +00:00
|
|
|
_currentRoom._gates.clear();
|
2009-07-15 19:06:24 +00:00
|
|
|
for (uint i = 0; i < _currentRoom._numGates; ++i) {
|
2009-07-31 04:32:33 +00:00
|
|
|
_currentRoom._gates.push_back(roomReader.readSint16LE());
|
2009-07-15 19:06:24 +00:00
|
|
|
}
|
|
|
|
|
2009-10-30 00:52:05 +00:00
|
|
|
// Load the walking map
|
2009-10-30 02:15:41 +00:00
|
|
|
loadWalkingMap(getMapID());
|
2009-10-30 00:52:05 +00:00
|
|
|
|
2009-11-01 12:57:06 +00:00
|
|
|
// Add overlays for the walking map and shortest/obliqued paths.
|
|
|
|
Animation *map = _vm->_anims->addAnimation(kWalkingMapOverlay, 256, _vm->_showWalkingMap);
|
|
|
|
Sprite *ov = _walkingMap.newOverlayFromMap(kWalkingMapOverlayColour);
|
|
|
|
map->addFrame(ov, NULL);
|
|
|
|
|
|
|
|
Animation *sPath = _vm->_anims->addAnimation(kWalkingShortestPathOverlay, 257, _vm->_showWalkingMap);
|
|
|
|
Animation *oPath = _vm->_anims->addAnimation(kWalkingObliquePathOverlay, 258, _vm->_showWalkingMap);
|
|
|
|
WalkingMap::Path emptyPath;
|
|
|
|
ov = _walkingMap.newOverlayFromPath(emptyPath, 0);
|
|
|
|
sPath->addFrame(ov, NULL);
|
|
|
|
ov = _walkingMap.newOverlayFromPath(emptyPath, 0);
|
|
|
|
oPath->addFrame(ov, NULL);
|
|
|
|
|
2009-07-18 03:05:20 +00:00
|
|
|
// Load the room's objects
|
2009-07-14 00:41:17 +00:00
|
|
|
for (uint i = 0; i < _info._numObjects; ++i) {
|
2009-09-30 16:04:21 +00:00
|
|
|
debugC(7, kDraciLogicDebugLevel,
|
2009-07-06 19:50:59 +00:00
|
|
|
"Checking if object %d (%d) is at the current location (%d)", i,
|
|
|
|
_objects[i]._location, roomNum);
|
|
|
|
|
|
|
|
if (_objects[i]._location == roomNum) {
|
2009-07-15 19:06:24 +00:00
|
|
|
debugC(6, kDraciLogicDebugLevel, "Loading object %d from room %d", i, roomNum);
|
2009-07-06 19:50:59 +00:00
|
|
|
loadObject(i);
|
|
|
|
}
|
|
|
|
}
|
2009-09-30 10:45:14 +00:00
|
|
|
|
|
|
|
// Run the init scripts for room objects
|
2009-07-06 19:50:59 +00:00
|
|
|
// We can't do this in the above loop because some objects' scripts reference
|
|
|
|
// other objects that may not yet be loaded
|
2009-07-14 00:41:17 +00:00
|
|
|
for (uint i = 0; i < _info._numObjects; ++i) {
|
2009-07-06 19:50:59 +00:00
|
|
|
if (_objects[i]._location == roomNum) {
|
2009-09-28 04:09:03 +00:00
|
|
|
const GameObject *obj = getObject(i);
|
2009-09-30 16:04:21 +00:00
|
|
|
debugC(6, kDraciLogicDebugLevel,
|
2009-09-30 10:45:14 +00:00
|
|
|
"Running init program for object %d (offset %d)", i, obj->_init);
|
2009-09-28 04:09:03 +00:00
|
|
|
_vm->_script->run(obj->_program, obj->_init);
|
2009-07-06 19:50:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-18 03:05:20 +00:00
|
|
|
// Load the room's GPL program and run the init part
|
2009-07-06 19:50:59 +00:00
|
|
|
f = _vm->_roomsArchive->getFile(roomNum * 4 + 3);
|
2009-07-17 01:20:51 +00:00
|
|
|
_currentRoom._program._bytecode = f->_data;
|
2009-07-06 19:50:59 +00:00
|
|
|
_currentRoom._program._length = f->_length;
|
|
|
|
|
2009-07-15 19:06:24 +00:00
|
|
|
debugC(4, kDraciLogicDebugLevel, "Running room init program...");
|
2009-07-06 19:50:59 +00:00
|
|
|
_vm->_script->run(_currentRoom._program, _currentRoom._init);
|
|
|
|
|
2009-07-18 03:05:20 +00:00
|
|
|
// Set room palette
|
2009-07-04 15:21:12 +00:00
|
|
|
f = _vm->_paletteArchive->getFile(_currentRoom._palette);
|
|
|
|
_vm->_screen->setPalette(f->_data, 0, kNumColours);
|
|
|
|
}
|
|
|
|
|
2009-07-07 21:18:28 +00:00
|
|
|
int Game::loadAnimation(uint animNum, uint z) {
|
2009-09-28 03:51:23 +00:00
|
|
|
const BAFile *animFile = _vm->_animationsArchive->getFile(animNum);
|
2009-09-30 16:04:21 +00:00
|
|
|
Common::MemoryReadStream animationReader(animFile->_data, animFile->_length);
|
2009-07-04 15:21:12 +00:00
|
|
|
|
2009-07-13 19:53:53 +00:00
|
|
|
uint numFrames = animationReader.readByte();
|
2009-07-04 15:21:12 +00:00
|
|
|
|
|
|
|
// FIXME: handle these properly
|
|
|
|
animationReader.readByte(); // Memory logic field, not used
|
|
|
|
animationReader.readByte(); // Disable erasing field, not used
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-07-07 21:30:36 +00:00
|
|
|
bool cyclic = animationReader.readByte();
|
|
|
|
|
2009-07-04 15:21:12 +00:00
|
|
|
animationReader.readByte(); // Relative field, not used
|
|
|
|
|
2009-07-07 21:18:28 +00:00
|
|
|
Animation *anim = _vm->_anims->addAnimation(animNum, z, false);
|
2009-07-12 19:32:01 +00:00
|
|
|
|
|
|
|
anim->setLooping(cyclic);
|
|
|
|
|
2009-07-04 15:21:12 +00:00
|
|
|
for (uint i = 0; i < numFrames; ++i) {
|
|
|
|
uint spriteNum = animationReader.readUint16LE() - 1;
|
|
|
|
int x = animationReader.readSint16LE();
|
|
|
|
int y = animationReader.readSint16LE();
|
2009-07-22 04:42:33 +00:00
|
|
|
uint scaledWidth = animationReader.readUint16LE();
|
|
|
|
uint scaledHeight = animationReader.readUint16LE();
|
2009-07-04 15:21:12 +00:00
|
|
|
byte mirror = animationReader.readByte();
|
2009-10-12 22:27:23 +00:00
|
|
|
int sample = animationReader.readUint16LE() - 1;
|
2009-10-11 23:01:59 +00:00
|
|
|
uint freq = animationReader.readUint16LE();
|
2009-07-04 15:21:12 +00:00
|
|
|
uint delay = animationReader.readUint16LE();
|
|
|
|
|
2009-11-01 10:15:48 +00:00
|
|
|
// _spritesArchive is flushed when entering a room. All
|
|
|
|
// scripts in a room are responsible for loading their animations.
|
2009-09-28 03:51:23 +00:00
|
|
|
const BAFile *spriteFile = _vm->_spritesArchive->getFile(spriteNum);
|
2009-07-04 18:35:08 +00:00
|
|
|
Sprite *sp = new Sprite(spriteFile->_data, spriteFile->_length, x, y, true);
|
2009-07-04 15:21:12 +00:00
|
|
|
|
2009-07-22 05:00:11 +00:00
|
|
|
// Some frames set the scaled dimensions to 0 even though other frames
|
|
|
|
// from the same animations have them set to normal values
|
|
|
|
// We work around this by assuming it means no scaling is necessary
|
|
|
|
if (scaledWidth == 0) {
|
|
|
|
scaledWidth = sp->getWidth();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (scaledHeight == 0) {
|
|
|
|
scaledHeight = sp->getHeight();
|
|
|
|
}
|
|
|
|
|
2009-07-22 04:42:33 +00:00
|
|
|
sp->setScaled(scaledWidth, scaledHeight);
|
|
|
|
|
2009-09-30 16:04:21 +00:00
|
|
|
if (mirror)
|
2009-07-04 15:21:12 +00:00
|
|
|
sp->setMirrorOn();
|
|
|
|
|
2009-07-12 19:32:01 +00:00
|
|
|
sp->setDelay(delay * 10);
|
2009-07-05 03:24:46 +00:00
|
|
|
|
2009-10-11 23:01:59 +00:00
|
|
|
const SoundSample *sam = _vm->_soundsArchive->getSample(sample, freq);
|
|
|
|
|
|
|
|
anim->addFrame(sp, sam);
|
2009-07-04 15:21:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return animNum;
|
|
|
|
}
|
2009-07-02 16:15:32 +00:00
|
|
|
|
2009-07-04 15:21:12 +00:00
|
|
|
void Game::loadObject(uint objNum) {
|
2009-09-28 03:51:23 +00:00
|
|
|
const BAFile *file;
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-07-02 16:15:32 +00:00
|
|
|
file = _vm->_objectsArchive->getFile(objNum * 3);
|
2009-06-28 13:10:53 +00:00
|
|
|
Common::MemoryReadStream objReader(file->_data, file->_length);
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-07-02 16:15:32 +00:00
|
|
|
GameObject *obj = _objects + objNum;
|
2009-07-02 15:08:42 +00:00
|
|
|
|
2009-06-28 13:10:53 +00:00
|
|
|
obj->_init = objReader.readUint16LE();
|
|
|
|
obj->_look = objReader.readUint16LE();
|
|
|
|
obj->_use = objReader.readUint16LE();
|
|
|
|
obj->_canUse = objReader.readUint16LE();
|
|
|
|
obj->_imInit = objReader.readByte();
|
|
|
|
obj->_imLook = objReader.readByte();
|
|
|
|
obj->_imUse = objReader.readByte();
|
2009-08-09 04:09:24 +00:00
|
|
|
obj->_walkDir = objReader.readByte() - 1;
|
2009-07-16 18:39:39 +00:00
|
|
|
obj->_z = objReader.readByte();
|
2009-07-07 21:18:28 +00:00
|
|
|
objReader.readUint16LE(); // idxSeq field, not used
|
|
|
|
objReader.readUint16LE(); // numSeq field, not used
|
2009-06-28 13:10:53 +00:00
|
|
|
obj->_lookX = objReader.readUint16LE();
|
|
|
|
obj->_lookY = objReader.readUint16LE();
|
|
|
|
obj->_useX = objReader.readUint16LE();
|
|
|
|
obj->_useY = objReader.readUint16LE();
|
2009-10-12 03:08:28 +00:00
|
|
|
obj->_lookDir = static_cast<SightDirection> (objReader.readByte());
|
|
|
|
obj->_useDir = static_cast<SightDirection> (objReader.readByte());
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-06-28 13:10:53 +00:00
|
|
|
obj->_absNum = objNum;
|
2009-09-30 10:45:14 +00:00
|
|
|
|
2009-07-02 16:15:32 +00:00
|
|
|
file = _vm->_objectsArchive->getFile(objNum * 3 + 1);
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-07-27 04:51:34 +00:00
|
|
|
// The first byte of the file is the length of the string (without the length)
|
|
|
|
assert(file->_length - 1 == file->_data[0]);
|
|
|
|
|
|
|
|
obj->_title = Common::String((char *)(file->_data+1), file->_length-1);
|
|
|
|
|
2009-07-02 16:15:32 +00:00
|
|
|
file = _vm->_objectsArchive->getFile(objNum * 3 + 2);
|
2009-07-17 01:20:51 +00:00
|
|
|
obj->_program._bytecode = file->_data;
|
2009-06-28 16:19:10 +00:00
|
|
|
obj->_program._length = file->_length;
|
2009-06-28 13:10:53 +00:00
|
|
|
}
|
2009-07-02 15:08:42 +00:00
|
|
|
|
2009-08-09 04:09:24 +00:00
|
|
|
void Game::loadWalkingMap(int mapID) {
|
2009-09-28 03:51:23 +00:00
|
|
|
const BAFile *f;
|
2009-08-09 04:09:24 +00:00
|
|
|
f = _vm->_walkingMapsArchive->getFile(mapID);
|
2009-10-30 02:15:41 +00:00
|
|
|
_walkingMap.load(f->_data, f->_length);
|
2009-08-09 04:09:24 +00:00
|
|
|
}
|
|
|
|
|
2009-07-04 15:21:12 +00:00
|
|
|
void Game::loadOverlays() {
|
2009-09-30 10:45:14 +00:00
|
|
|
uint x, y, z, num;
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-09-28 03:51:23 +00:00
|
|
|
const BAFile *overlayHeader;
|
2009-07-04 15:21:12 +00:00
|
|
|
|
2009-10-01 08:32:35 +00:00
|
|
|
overlayHeader = _vm->_roomsArchive->getFile(getRoomNum() * 4 + 2);
|
2009-07-04 15:21:12 +00:00
|
|
|
Common::MemoryReadStream overlayReader(overlayHeader->_data, overlayHeader->_length);
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-09-30 10:45:14 +00:00
|
|
|
for (int i = 0; i < _currentRoom._numOverlays; i++) {
|
2009-07-04 15:21:12 +00:00
|
|
|
num = overlayReader.readUint16LE() - 1;
|
2009-09-30 10:45:14 +00:00
|
|
|
x = overlayReader.readUint16LE();
|
|
|
|
y = overlayReader.readUint16LE();
|
|
|
|
z = overlayReader.readByte();
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-11-01 10:15:48 +00:00
|
|
|
// _overlaysArchive is flushed when entering a room and this
|
|
|
|
// code is called after the flushing has been done.
|
2009-09-28 03:51:23 +00:00
|
|
|
const BAFile *overlayFile;
|
2009-07-04 15:21:12 +00:00
|
|
|
overlayFile = _vm->_overlaysArchive->getFile(num);
|
2009-09-30 10:45:14 +00:00
|
|
|
Sprite *sp = new Sprite(overlayFile->_data, overlayFile->_length, x, y, true);
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-09-30 10:45:14 +00:00
|
|
|
_vm->_anims->addOverlay(sp, z);
|
2009-07-03 17:54:13 +00:00
|
|
|
}
|
2009-07-04 15:21:12 +00:00
|
|
|
|
2009-09-30 10:45:14 +00:00
|
|
|
_vm->_screen->getSurface()->markDirty();
|
2009-07-04 15:21:12 +00:00
|
|
|
}
|
|
|
|
|
2009-10-04 22:11:46 +00:00
|
|
|
void Game::deleteObjectAnimations() {
|
|
|
|
for (uint i = 0; i < _info._numObjects; ++i) {
|
|
|
|
GameObject *obj = &_objects[i];
|
|
|
|
|
|
|
|
if (i != 0 && (obj->_location == getPreviousRoomNum())) {
|
2009-10-12 00:01:39 +00:00
|
|
|
for (uint j = 0; j < obj->_anim.size(); ++j) {
|
|
|
|
_vm->_anims->deleteAnimation(obj->_anim[j]);
|
2009-10-04 22:11:46 +00:00
|
|
|
}
|
2009-10-12 00:01:39 +00:00
|
|
|
obj->_anim.clear();
|
2009-10-04 22:11:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-10-12 01:16:13 +00:00
|
|
|
int Game::playingObjectAnimation(const GameObject *obj) const {
|
|
|
|
for (uint i = 0; i < obj->_anim.size(); ++i) {
|
|
|
|
const int animID = obj->_anim[i];
|
|
|
|
const Animation *anim = _vm->_anims->getAnimation(animID);
|
|
|
|
if (anim && anim->isPlaying()) {
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2009-10-04 22:11:46 +00:00
|
|
|
void Game::enterNewRoom(bool force_reload) {
|
|
|
|
if (_newRoom == getRoomNum() && !force_reload) {
|
2009-10-01 08:32:35 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
debugC(1, kDraciLogicDebugLevel, "Entering room %d using gate %d", _newRoom, _newGate);
|
2009-07-14 19:13:49 +00:00
|
|
|
|
2009-10-12 22:27:23 +00:00
|
|
|
// TODO: maybe wait till all sounds end instead of stopping them.
|
|
|
|
// In any case, make sure all sounds are stopped before we deallocate
|
|
|
|
// their memory by clearing the cache.
|
|
|
|
_vm->_sound->stopAll();
|
|
|
|
|
2009-07-17 01:20:51 +00:00
|
|
|
// Clear archives
|
2009-07-08 00:40:22 +00:00
|
|
|
_vm->_roomsArchive->clearCache();
|
|
|
|
_vm->_spritesArchive->clearCache();
|
|
|
|
_vm->_paletteArchive->clearCache();
|
2009-07-17 01:20:51 +00:00
|
|
|
_vm->_animationsArchive->clearCache();
|
|
|
|
_vm->_walkingMapsArchive->clearCache();
|
2009-10-11 23:01:59 +00:00
|
|
|
_vm->_soundsArchive->clearCache();
|
|
|
|
_vm->_dubbingArchive->clearCache();
|
2009-11-01 10:15:48 +00:00
|
|
|
_vm->_overlaysArchive->clearCache();
|
2009-07-08 00:40:22 +00:00
|
|
|
|
2009-07-12 19:00:24 +00:00
|
|
|
_vm->_screen->clearScreen();
|
|
|
|
|
2009-07-07 15:21:41 +00:00
|
|
|
_vm->_anims->deleteOverlays();
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-07-18 03:00:12 +00:00
|
|
|
// Delete walking map testing overlay
|
2009-07-25 18:29:43 +00:00
|
|
|
_vm->_anims->deleteAnimation(kWalkingMapOverlay);
|
2009-11-01 12:57:06 +00:00
|
|
|
_vm->_anims->deleteAnimation(kWalkingShortestPathOverlay);
|
|
|
|
_vm->_anims->deleteAnimation(kWalkingObliquePathOverlay);
|
2009-07-07 15:21:41 +00:00
|
|
|
|
2009-07-18 03:12:12 +00:00
|
|
|
// TODO: Make objects capable of stopping their own animations
|
2009-09-28 04:09:03 +00:00
|
|
|
const GameObject *dragon = getObject(kDragonObject);
|
2009-10-12 00:15:34 +00:00
|
|
|
stopObjectAnimations(dragon);
|
2009-07-18 03:12:12 +00:00
|
|
|
|
2009-10-01 08:32:35 +00:00
|
|
|
// Remember the previous room for returning back from the map.
|
2009-10-04 22:11:46 +00:00
|
|
|
rememberRoomNumAsPrevious();
|
|
|
|
deleteObjectAnimations();
|
2009-07-07 15:21:41 +00:00
|
|
|
|
2009-10-01 08:32:35 +00:00
|
|
|
// Set the current room to the new value
|
|
|
|
_currentRoom._roomNum = _newRoom;
|
|
|
|
|
|
|
|
// Before setting these variables we have to convert the values to 1-based indexing
|
|
|
|
// because this is how everything is stored in the data files
|
|
|
|
_variables[0] = _newGate + 1;
|
|
|
|
_variables[1] = _newRoom + 1;
|
|
|
|
|
|
|
|
// If the new room is the map room, set the appropriate coordinates
|
|
|
|
// for the dragon in the persons array
|
|
|
|
if (_newRoom == _info._mapRoom) {
|
|
|
|
_persons[kDragonObject]._x = 160;
|
|
|
|
_persons[kDragonObject]._y = 0;
|
|
|
|
}
|
|
|
|
|
2009-10-01 16:47:34 +00:00
|
|
|
// Set the appropriate loop statu before loading the room
|
|
|
|
setLoopStatus(kStatusGate);
|
|
|
|
setLoopSubstatus(kSubstatusOrdinary);
|
|
|
|
|
2009-10-01 08:32:35 +00:00
|
|
|
loadRoom(_newRoom);
|
2009-07-06 19:50:59 +00:00
|
|
|
loadOverlays();
|
2009-10-01 08:32:35 +00:00
|
|
|
|
|
|
|
// Run the program for the gate the dragon came through
|
|
|
|
runGateProgram(_newGate);
|
|
|
|
|
|
|
|
// Set cursor state
|
|
|
|
// Need to do this after we set the palette since the cursors use it
|
|
|
|
if (_currentRoom._mouseOn) {
|
|
|
|
debugC(6, kDraciLogicDebugLevel, "Mouse: ON");
|
|
|
|
_vm->_mouse->cursorOn();
|
|
|
|
} else {
|
|
|
|
debugC(6, kDraciLogicDebugLevel, "Mouse: OFF");
|
|
|
|
_vm->_mouse->cursorOff();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Reset the loop status.
|
|
|
|
setLoopStatus(kStatusOrdinary);
|
2009-10-28 00:20:45 +00:00
|
|
|
|
|
|
|
_vm->_mouse->setCursorType(kNormalCursor);
|
2009-07-06 19:50:59 +00:00
|
|
|
}
|
|
|
|
|
2009-07-31 04:32:33 +00:00
|
|
|
void Game::runGateProgram(int gate) {
|
2009-08-01 17:52:16 +00:00
|
|
|
debugC(6, kDraciLogicDebugLevel, "Running program for gate %d", gate);
|
|
|
|
|
2009-07-31 04:32:33 +00:00
|
|
|
// Mark last animation
|
|
|
|
int lastAnimIndex = _vm->_anims->getLastIndex();
|
|
|
|
|
|
|
|
// Run gate program
|
|
|
|
_vm->_script->run(_currentRoom._program, _currentRoom._gates[gate]);
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-09-27 21:25:34 +00:00
|
|
|
deleteAnimationsAfterIndex(lastAnimIndex);
|
2009-07-31 04:32:33 +00:00
|
|
|
|
|
|
|
setExitLoop(false);
|
|
|
|
}
|
|
|
|
|
2009-09-28 23:10:08 +00:00
|
|
|
void Game::positionAnimAsHero(Animation *anim) {
|
|
|
|
// Calculate scaling factors
|
2009-09-29 05:54:59 +00:00
|
|
|
const double scale = getPers0() + getPersStep() * _hero.y;
|
2009-09-28 23:10:08 +00:00
|
|
|
|
|
|
|
// Set the Z coordinate for the dragon's animation
|
2009-09-29 05:54:59 +00:00
|
|
|
anim->setZ(_hero.y + 1);
|
2009-09-28 23:10:08 +00:00
|
|
|
|
|
|
|
// Fetch current frame
|
2009-10-30 01:56:52 +00:00
|
|
|
Drawable *frame = anim->getCurrentFrame();
|
2009-09-28 23:10:08 +00:00
|
|
|
|
|
|
|
// Fetch base dimensions of the frame
|
|
|
|
uint height = frame->getHeight();
|
|
|
|
uint width = frame->getWidth();
|
|
|
|
|
|
|
|
// We naturally want the dragon to position its feet to the location of the
|
|
|
|
// click but sprites are drawn from their top-left corner so we subtract
|
|
|
|
// the current height of the dragon's sprite
|
2009-09-29 05:54:59 +00:00
|
|
|
Common::Point p = _hero;
|
|
|
|
p.x -= (int)(scale * width) / 2;
|
|
|
|
p.y -= (int)(scale * height);
|
2009-09-28 23:10:08 +00:00
|
|
|
|
2009-09-29 05:54:59 +00:00
|
|
|
// Since _persons[] is used for placing talking text, we use the non-adjusted x value
|
|
|
|
// so the text remains centered over the dragon.
|
2009-09-30 10:45:14 +00:00
|
|
|
_persons[kDragonObject]._x = _hero.x;
|
2009-09-29 05:54:59 +00:00
|
|
|
_persons[kDragonObject]._y = p.y;
|
|
|
|
|
|
|
|
// Set the per-animation scaling factor
|
|
|
|
anim->setScaleFactors(scale, scale);
|
|
|
|
|
|
|
|
anim->setRelative(p.x, p.y);
|
2009-09-28 23:10:08 +00:00
|
|
|
}
|
|
|
|
|
2009-10-11 23:28:40 +00:00
|
|
|
void Game::pushNewRoom() {
|
|
|
|
_pushedNewRoom = _newRoom;
|
|
|
|
_pushedNewGate = _newGate;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Game::popNewRoom() {
|
|
|
|
if (_loopStatus != kStatusInventory && _pushedNewRoom >= 0) {
|
|
|
|
scheduleEnteringRoomUsingGate(_pushedNewRoom, _pushedNewGate);
|
|
|
|
_pushedNewRoom = _pushedNewGate = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-10-13 04:44:22 +00:00
|
|
|
void Game::setSpeechTiming(uint tick, uint duration) {
|
2009-08-02 05:21:21 +00:00
|
|
|
_speechTick = tick;
|
2009-10-13 04:44:22 +00:00
|
|
|
_speechDuration = duration;
|
2009-08-02 05:21:21 +00:00
|
|
|
}
|
|
|
|
|
2009-10-28 07:34:17 +00:00
|
|
|
void Game::shiftSpeechAndFadeTick(int delta) {
|
2009-10-13 05:38:45 +00:00
|
|
|
_speechTick += delta;
|
2009-10-28 07:34:17 +00:00
|
|
|
_fadeTick += delta;
|
2009-10-13 05:38:45 +00:00
|
|
|
}
|
|
|
|
|
2009-10-28 07:34:17 +00:00
|
|
|
void Game::initializeFading(int phases) {
|
|
|
|
_fadePhases = _fadePhase = phases;
|
|
|
|
_fadeTick = _vm->_system->getMillis();
|
|
|
|
}
|
|
|
|
|
2009-09-27 21:25:34 +00:00
|
|
|
void Game::deleteAnimationsAfterIndex(int lastAnimIndex) {
|
2009-09-30 16:04:21 +00:00
|
|
|
// Delete all animations loaded after the marked one
|
2009-09-27 21:25:34 +00:00
|
|
|
// (from objects and from the AnimationManager)
|
|
|
|
for (uint i = 0; i < getNumObjects(); ++i) {
|
|
|
|
GameObject *obj = &_objects[i];
|
|
|
|
|
2009-10-12 00:01:39 +00:00
|
|
|
for (uint j = 0; j < obj->_anim.size(); ++j) {
|
2009-09-27 21:25:34 +00:00
|
|
|
Animation *anim;
|
|
|
|
|
2009-10-12 00:01:39 +00:00
|
|
|
anim = _vm->_anims->getAnimation(obj->_anim[j]);
|
2009-09-27 21:25:34 +00:00
|
|
|
if (anim != NULL && anim->getIndex() > lastAnimIndex)
|
2009-10-12 00:01:39 +00:00
|
|
|
obj->_anim.remove_at(j--);
|
2009-09-27 21:25:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_vm->_anims->deleteAfterIndex(lastAnimIndex);
|
|
|
|
}
|
2009-10-12 00:15:34 +00:00
|
|
|
|
|
|
|
void Game::stopObjectAnimations(const GameObject *obj) {
|
|
|
|
for (uint i = 0; i < obj->_anim.size(); ++i) {
|
|
|
|
_vm->_anims->stop(obj->_anim[i]);
|
|
|
|
}
|
|
|
|
}
|
2009-09-27 21:25:34 +00:00
|
|
|
|
2009-06-27 05:10:24 +00:00
|
|
|
Game::~Game() {
|
|
|
|
delete[] _persons;
|
|
|
|
delete[] _variables;
|
2009-08-11 04:03:22 +00:00
|
|
|
delete[] _dialogueOffsets;
|
2009-09-28 11:28:09 +00:00
|
|
|
delete[] _dialogueVars;
|
2009-07-02 15:08:42 +00:00
|
|
|
delete[] _objects;
|
2009-09-28 03:51:23 +00:00
|
|
|
delete[] _itemStatus;
|
2009-08-17 18:50:38 +00:00
|
|
|
delete[] _items;
|
2009-06-28 13:10:53 +00:00
|
|
|
}
|
|
|
|
|
2009-10-04 05:44:23 +00:00
|
|
|
void Game::DoSync(Common::Serializer &s) {
|
|
|
|
s.syncAsUint16LE(_currentRoom._roomNum);
|
|
|
|
|
|
|
|
for (uint i = 0; i < _info._numObjects; ++i) {
|
|
|
|
GameObject& obj = _objects[i];
|
|
|
|
s.syncAsSint16LE(obj._location);
|
|
|
|
s.syncAsByte(obj._visible);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint i = 0; i < _info._numItems; ++i) {
|
|
|
|
s.syncAsByte(_itemStatus[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < kInventorySlots; ++i) {
|
|
|
|
s.syncAsSint16LE(_inventory[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < _info._numVariables; ++i) {
|
|
|
|
s.syncAsSint16LE(_variables[i]);
|
|
|
|
}
|
|
|
|
for (uint i = 0; i < _info._numDialogueBlocks; ++i) {
|
|
|
|
s.syncAsSint16LE(_dialogueVars[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2009-07-29 01:11:53 +00:00
|
|
|
|
2009-07-19 13:28:05 +00:00
|
|
|
static double real_to_double(byte real[6]) {
|
|
|
|
// Extract sign bit
|
|
|
|
int sign = real[0] & (1 << 7);
|
2009-09-30 16:04:21 +00:00
|
|
|
|
2009-07-19 13:28:05 +00:00
|
|
|
// Extract exponent and adjust for bias
|
|
|
|
int exp = real[5] - 129;
|
|
|
|
|
|
|
|
double mantissa;
|
|
|
|
double tmp = 0.0;
|
|
|
|
|
|
|
|
if (real[5] == 0) {
|
|
|
|
mantissa = 0.0;
|
|
|
|
} else {
|
|
|
|
// Process the first four least significant bytes
|
2009-09-30 10:45:14 +00:00
|
|
|
for (int i = 4; i >= 1; --i) {
|
2009-07-19 13:28:05 +00:00
|
|
|
tmp += real[i];
|
|
|
|
tmp /= 1 << 8;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Process the most significant byte (remove the sign bit)
|
|
|
|
tmp += real[0] & ((1 << 7) - 1);
|
|
|
|
tmp /= 1 << 8;
|
|
|
|
|
|
|
|
// Calculate mantissa
|
|
|
|
mantissa = 1.0;
|
|
|
|
mantissa += 2.0 * tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Flip sign if necessary
|
|
|
|
if (sign) {
|
|
|
|
mantissa = -mantissa;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Calculate final value
|
2009-09-30 10:45:14 +00:00
|
|
|
return ldexp(mantissa, exp);
|
2009-07-19 13:28:05 +00:00
|
|
|
}
|
2009-07-18 03:00:12 +00:00
|
|
|
|
2009-09-30 16:04:21 +00:00
|
|
|
}
|