mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-09 11:20:56 +00:00
1199 lines
32 KiB
C++
1199 lines
32 KiB
C++
/* ScummVM - Graphic Adventure Engine
|
|
*
|
|
* ScummVM is the legal property of its developers, whose names
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
* file distributed with this source distribution.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License
|
|
* as published by the Free Software Foundation; either version 2
|
|
* of the License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
*/
|
|
|
|
/*
|
|
* This code is based on Labyrinth of Time code with assistance of
|
|
*
|
|
* Copyright (c) 1993 Terra Nova Development
|
|
* Copyright (c) 2004 The Wyrmkeep Entertainment Co.
|
|
*
|
|
*/
|
|
|
|
#include "lab/lab.h"
|
|
#include "lab/labfun.h"
|
|
#include "lab/anim.h"
|
|
#include "lab/image.h"
|
|
#include "lab/text.h"
|
|
#include "lab/intro.h"
|
|
#include "lab/parsefun.h"
|
|
#include "lab/interface.h"
|
|
#include "lab/resource.h"
|
|
|
|
namespace Lab {
|
|
|
|
extern bool DoNotDrawMessage;
|
|
|
|
/* Global parser data */
|
|
|
|
extern RoomData *_rooms;
|
|
extern uint16 NumInv, ManyRooms, HighestCondition, Direction;
|
|
|
|
bool ispal = false, noupdatediff = false, MainDisplay = true, QuitLab = false;
|
|
|
|
/* LAB: Labyrinth specific code for the special puzzles */
|
|
#define SPECIALLOCK 100
|
|
#define SPECIALBRICK 101
|
|
#define SPECIALBRICKNOMOUSE 102
|
|
|
|
#define MAPNUM 28
|
|
#define JOURNALNUM 9
|
|
#define WESTPAPERNUM 18
|
|
#define NOTESNUM 12
|
|
#define WHISKEYNUM 25
|
|
#define PITHHELMETNUM 7
|
|
#define HELMETNUM 1
|
|
|
|
#define LAMPNUM 27
|
|
#define LAMPON 151
|
|
|
|
#define BELTNUM 3
|
|
#define BELTGLOW 70
|
|
|
|
#define USEDHELMET 184
|
|
|
|
#define QUARTERNUM 30
|
|
|
|
|
|
#define MUSEUMMONITOR 71
|
|
#define GRAMAPHONEMONITOR 72
|
|
#define UNICYCLEMONITOR 73
|
|
#define STATUEMONITOR 74
|
|
#define TALISMANMONITOR 75
|
|
#define LUTEMONITOR 76
|
|
#define CLOCKMONITOR 77
|
|
#define WINDOWMONITOR 78
|
|
#define BELTMONITOR 79
|
|
#define LIBRARYMONITOR 80
|
|
#define TERMINALMONITOR 81
|
|
#define LEVERSMONITOR 82
|
|
|
|
static char initcolors[] = { '\x00', '\x00', '\x00', '\x30',
|
|
'\x30', '\x30', '\x10', '\x10',
|
|
'\x10', '\x14', '\x14', '\x14',
|
|
'\x20', '\x20', '\x20', '\x24',
|
|
'\x24', '\x24', '\x2c', '\x2c',
|
|
'\x2c', '\x08', '\x08', '\x08'};
|
|
|
|
/******************************************************************************/
|
|
/* Draws the message for the room. */
|
|
/******************************************************************************/
|
|
void LabEngine::drawRoomMessage(uint16 curInv, CloseDataPtr closePtr) {
|
|
if (_lastTooLong) {
|
|
_lastTooLong = false;
|
|
return;
|
|
}
|
|
|
|
if (_alternate) {
|
|
if ((curInv <= NumInv) && _conditions->in(curInv) && _inventory[curInv].BInvName) {
|
|
if ((curInv == LAMPNUM) && _conditions->in(LAMPON)) /* LAB: Labyrinth specific */
|
|
drawStaticMessage(kTextLampOn);
|
|
else if (_inventory[curInv].Many > 1) {
|
|
Common::String roomMessage = Common::String(_inventory[curInv].name) + " (" + Common::String::format("%d", _inventory[curInv].Many) + ")";
|
|
_graphics->drawMessage(roomMessage.c_str());
|
|
} else
|
|
_graphics->drawMessage(_inventory[curInv].name);
|
|
}
|
|
} else
|
|
drawDirection(closePtr);
|
|
|
|
_lastTooLong = _graphics->_lastMessageLong;
|
|
}
|
|
|
|
void LabEngine::freeScreens() {
|
|
for (uint16 i = 0; i < 20; i++) {
|
|
delete _moveImages[i];
|
|
_moveImages[i] = nullptr;
|
|
}
|
|
|
|
for (uint16 imgIdx = 0; imgIdx < 10; imgIdx++) {
|
|
delete _invImages[imgIdx];
|
|
delete Images[imgIdx];
|
|
_invImages[imgIdx] = Images[imgIdx] = nullptr;
|
|
}
|
|
}
|
|
|
|
|
|
/******************************************************************************/
|
|
/* Permanently flips the imagery of a gadget. */
|
|
/******************************************************************************/
|
|
void LabEngine::perFlipGadget(uint16 gadgetId) {
|
|
Gadget *topGadget = _moveGadgetList;
|
|
|
|
while (topGadget) {
|
|
if (topGadget->GadgetID == gadgetId) {
|
|
Image *tmpImage = topGadget->_image;
|
|
topGadget->_image = topGadget->_altImage;
|
|
topGadget->_altImage = tmpImage;
|
|
|
|
if (!_alternate) {
|
|
_event->mouseHide();
|
|
topGadget->_image->drawImage(topGadget->x, topGadget->y);
|
|
_event->mouseShow();
|
|
}
|
|
|
|
return;
|
|
} else
|
|
topGadget = topGadget->NextGadget;
|
|
}
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/* Eats all the available messages. */
|
|
/******************************************************************************/
|
|
void LabEngine::eatMessages() {
|
|
IntuiMessage *msg;
|
|
|
|
do {
|
|
msg = getMsg();
|
|
} while (msg);
|
|
|
|
return;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/* Checks whether the close up is one of the special case closeups. */
|
|
/******************************************************************************/
|
|
bool LabEngine::doCloseUp(CloseDataPtr closePtr) {
|
|
if (closePtr == NULL)
|
|
return false;
|
|
|
|
int monltmargin, monrtmargin, montopmargin, lutertmargin;
|
|
|
|
if (getPlatform() != Common::kPlatformWindows) {
|
|
monltmargin = 0;
|
|
monrtmargin = 319;
|
|
montopmargin = 0;
|
|
lutertmargin = 124;
|
|
} else {
|
|
monltmargin = 2;
|
|
monrtmargin = 317;
|
|
montopmargin = 2;
|
|
lutertmargin = 128;
|
|
}
|
|
|
|
switch (closePtr->CloseUpType) {
|
|
case MUSEUMMONITOR:
|
|
case LIBRARYMONITOR:
|
|
case WINDOWMONITOR:
|
|
doMonitor(closePtr->GraphicName, closePtr->Message, false, monltmargin, montopmargin, monrtmargin, 165);
|
|
break;
|
|
case GRAMAPHONEMONITOR:
|
|
doMonitor(closePtr->GraphicName, closePtr->Message, false, monltmargin, montopmargin, 171, 165);
|
|
break;
|
|
case UNICYCLEMONITOR:
|
|
doMonitor(closePtr->GraphicName, closePtr->Message, false, 100, montopmargin, monrtmargin, 165);
|
|
break;
|
|
case STATUEMONITOR:
|
|
doMonitor(closePtr->GraphicName, closePtr->Message, false, 117, montopmargin, monrtmargin, 165);
|
|
break;
|
|
case TALISMANMONITOR:
|
|
doMonitor(closePtr->GraphicName, closePtr->Message, false, monltmargin, montopmargin, 184, 165);
|
|
break;
|
|
case LUTEMONITOR:
|
|
doMonitor(closePtr->GraphicName, closePtr->Message, false, monltmargin, montopmargin, lutertmargin, 165);
|
|
break;
|
|
case CLOCKMONITOR:
|
|
doMonitor(closePtr->GraphicName, closePtr->Message, false, monltmargin, montopmargin, 206, 165);
|
|
break;
|
|
case TERMINALMONITOR:
|
|
doMonitor(closePtr->GraphicName, closePtr->Message, true, monltmargin, montopmargin, monrtmargin, 165);
|
|
break;
|
|
default:
|
|
return false;
|
|
}
|
|
|
|
_curFileName = " ";
|
|
_graphics->drawPanel();
|
|
|
|
return true;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/* Gets the current inventory name. */
|
|
/******************************************************************************/
|
|
const char *LabEngine::getInvName(uint16 CurInv) {
|
|
if (MainDisplay)
|
|
return _inventory[CurInv].BInvName;
|
|
|
|
if ((CurInv == LAMPNUM) && _conditions->in(LAMPON))
|
|
return "P:Mines/120";
|
|
|
|
if ((CurInv == BELTNUM) && _conditions->in(BELTGLOW))
|
|
return "P:Future/BeltGlow";
|
|
|
|
if (CurInv == WESTPAPERNUM) {
|
|
_curFileName = _inventory[CurInv].BInvName;
|
|
_anim->_noPalChange = true;
|
|
_graphics->readPict(_curFileName, false);
|
|
_anim->_noPalChange = false;
|
|
doWestPaper();
|
|
} else if (CurInv == NOTESNUM) {
|
|
_curFileName = _inventory[CurInv].BInvName;
|
|
_anim->_noPalChange = true;
|
|
_graphics->readPict(_curFileName, false);
|
|
_anim->_noPalChange = false;
|
|
doNotes();
|
|
}
|
|
|
|
return _inventory[CurInv].BInvName;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/* Turns the interface off. */
|
|
/******************************************************************************/
|
|
void LabEngine::interfaceOff() {
|
|
if (!_interfaceOff) {
|
|
_event->attachGadgetList(NULL);
|
|
_event->mouseHide();
|
|
_interfaceOff = true;
|
|
}
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/* Turns the interface on. */
|
|
/******************************************************************************/
|
|
void LabEngine::interfaceOn() {
|
|
if (_interfaceOff) {
|
|
_interfaceOff = false;
|
|
_event->mouseShow();
|
|
}
|
|
|
|
if (_graphics->_longWinInFront)
|
|
_event->attachGadgetList(NULL);
|
|
else if (_alternate)
|
|
_event->attachGadgetList(_invGadgetList);
|
|
else
|
|
_event->attachGadgetList(_moveGadgetList);
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/* If the user hits the "Use" gadget; things that can get used on themselves. */
|
|
/******************************************************************************/
|
|
bool LabEngine::doUse(uint16 CurInv) {
|
|
if (CurInv == MAPNUM) { /* LAB: Labyrinth specific */
|
|
drawStaticMessage(kTextUseMap);
|
|
interfaceOff();
|
|
_anim->stopDiff();
|
|
_curFileName = " ";
|
|
_cptr = NULL;
|
|
doMap(_roomNum);
|
|
setPalette(initcolors, 8);
|
|
_graphics->drawMessage(NULL);
|
|
_graphics->drawPanel();
|
|
} else if (CurInv == JOURNALNUM) { /* LAB: Labyrinth specific */
|
|
drawStaticMessage(kTextUseJournal);
|
|
interfaceOff();
|
|
_anim->stopDiff();
|
|
_curFileName = " ";
|
|
_cptr = NULL;
|
|
doJournal();
|
|
_graphics->drawPanel();
|
|
_graphics->drawMessage(NULL);
|
|
} else if (CurInv == LAMPNUM) { /* LAB: Labyrinth specific */
|
|
interfaceOff();
|
|
|
|
if (_conditions->in(LAMPON)) {
|
|
drawStaticMessage(kTextTurnLampOff);
|
|
_conditions->exclElement(LAMPON);
|
|
} else {
|
|
drawStaticMessage(kTextTurnLampOn);
|
|
_conditions->inclElement(LAMPON);
|
|
}
|
|
|
|
_anim->_doBlack = false;
|
|
_anim->_waitForEffect = true;
|
|
_graphics->readPict("Music:Click", true);
|
|
_anim->_waitForEffect = false;
|
|
|
|
_anim->_doBlack = false;
|
|
_nextFileName = getInvName(CurInv);
|
|
} else if (CurInv == BELTNUM) { /* LAB: Labyrinth specific */
|
|
if (!_conditions->in(BELTGLOW))
|
|
_conditions->inclElement(BELTGLOW);
|
|
|
|
_anim->_doBlack = false;
|
|
_nextFileName = getInvName(CurInv);
|
|
} else if (CurInv == WHISKEYNUM) { /* LAB: Labyrinth specific */
|
|
_conditions->inclElement(USEDHELMET);
|
|
drawStaticMessage(kTextUseWhiskey);
|
|
} else if (CurInv == PITHHELMETNUM) { /* LAB: Labyrinth specific */
|
|
_conditions->inclElement(USEDHELMET);
|
|
drawStaticMessage(kTextUsePith);
|
|
} else if (CurInv == HELMETNUM) { /* LAB: Labyrinth specific */
|
|
_conditions->inclElement(USEDHELMET);
|
|
drawStaticMessage(kTextUseHelmet);
|
|
} else
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/* Decrements the current inventory number. */
|
|
/******************************************************************************/
|
|
void LabEngine::decIncInv(uint16 *CurInv, bool dec) {
|
|
interfaceOff();
|
|
|
|
if (dec)
|
|
(*CurInv)--;
|
|
else
|
|
(*CurInv)++;
|
|
|
|
while (*CurInv && (*CurInv <= NumInv)) {
|
|
if (_conditions->in(*CurInv) && _inventory[*CurInv].BInvName) {
|
|
_nextFileName = getInvName(*CurInv);
|
|
break;
|
|
}
|
|
|
|
if (dec)
|
|
(*CurInv)--;
|
|
else
|
|
(*CurInv)++;
|
|
}
|
|
|
|
if ((*CurInv == 0) || (*CurInv > NumInv)) {
|
|
if (dec)
|
|
*CurInv = NumInv;
|
|
else
|
|
*CurInv = 1;
|
|
|
|
while (*CurInv && (*CurInv <= NumInv)) {
|
|
if (_conditions->in(*CurInv) && _inventory[*CurInv].BInvName) {
|
|
_nextFileName = getInvName(*CurInv);
|
|
break;
|
|
}
|
|
|
|
if (dec)
|
|
(*CurInv)--;
|
|
else
|
|
(*CurInv)++;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/******************************************************************************/
|
|
/* The main game loop */
|
|
/******************************************************************************/
|
|
void LabEngine::mainGameLoop() {
|
|
uint16 actionMode = 4;
|
|
uint16 curInv = MAPNUM;
|
|
|
|
bool forceDraw = false, GotMessage = true;
|
|
|
|
setPalette(initcolors, 8);
|
|
|
|
_cptr = NULL;
|
|
_roomNum = 1;
|
|
Direction = NORTH;
|
|
|
|
_resource->readRoomData("LAB:Doors");
|
|
if (!(_inventory = _resource->readInventory("LAB:Inventor")))
|
|
return;
|
|
|
|
if (!(_conditions = new LargeSet(HighestCondition + 1, this)))
|
|
return;
|
|
|
|
if (!(_roomsFound = new LargeSet(ManyRooms + 1, this)))
|
|
return;
|
|
|
|
_conditions->readInitialConditions("LAB:Conditio");
|
|
|
|
_graphics->_longWinInFront = false;
|
|
_graphics->drawPanel();
|
|
|
|
perFlipGadget(actionMode);
|
|
|
|
/* Set up initial picture. */
|
|
|
|
while (1) {
|
|
_event->processInput(true);
|
|
|
|
if (GotMessage) {
|
|
if (QuitLab || g_engine->shouldQuit()) {
|
|
_anim->stopDiff();
|
|
break;
|
|
}
|
|
|
|
_music->resumeBackMusic();
|
|
|
|
/* Sees what kind of close up we're in and does the appropriate stuff, if any. */
|
|
if (doCloseUp(_cptr)) {
|
|
_cptr = NULL;
|
|
|
|
mayShowCrumbIndicator();
|
|
_graphics->screenUpdate();
|
|
}
|
|
|
|
/* Sets the current picture properly on the screen */
|
|
if (MainDisplay)
|
|
_nextFileName = getPictName(&_cptr);
|
|
|
|
if (noupdatediff) {
|
|
_roomsFound->inclElement(_roomNum); /* Potentially entered another room */
|
|
forceDraw |= (strcmp(_nextFileName, _curFileName) != 0);
|
|
|
|
noupdatediff = false;
|
|
_curFileName = _nextFileName;
|
|
} else if (strcmp(_nextFileName, _curFileName) != 0) {
|
|
interfaceOff();
|
|
_roomsFound->inclElement(_roomNum); /* Potentially entered another room */
|
|
_curFileName = _nextFileName;
|
|
|
|
if (_cptr) {
|
|
if ((_cptr->CloseUpType == SPECIALLOCK) && MainDisplay) /* LAB: Labyrinth specific code */
|
|
showCombination(_curFileName);
|
|
else if (((_cptr->CloseUpType == SPECIALBRICK) ||
|
|
(_cptr->CloseUpType == SPECIALBRICKNOMOUSE)) &&
|
|
MainDisplay) /* LAB: Labyrinth specific code */
|
|
showTile(_curFileName, (bool)(_cptr->CloseUpType == SPECIALBRICKNOMOUSE));
|
|
else
|
|
_graphics->readPict(_curFileName, false);
|
|
} else
|
|
_graphics->readPict(_curFileName, false);
|
|
|
|
drawRoomMessage(curInv, _cptr);
|
|
forceDraw = false;
|
|
|
|
mayShowCrumbIndicator();
|
|
_graphics->screenUpdate();
|
|
|
|
if (!_followingCrumbs)
|
|
eatMessages();
|
|
}
|
|
|
|
if (forceDraw) {
|
|
drawRoomMessage(curInv, _cptr);
|
|
forceDraw = false;
|
|
_graphics->screenUpdate();
|
|
}
|
|
}
|
|
|
|
_music->updateMusic(); /* Make sure we check the music at least after every message */
|
|
interfaceOn();
|
|
IntuiMessage *curMsg = getMsg();
|
|
|
|
if (curMsg == NULL) { /* Does music load and next animation frame when you've run out of messages */
|
|
GotMessage = false;
|
|
_music->checkRoomMusic();
|
|
_music->updateMusic();
|
|
_anim->diffNextFrame();
|
|
|
|
if (_followingCrumbs) {
|
|
int result = followCrumbs();
|
|
|
|
if (result != 0) {
|
|
uint16 code = 0;
|
|
if (result == VKEY_UPARROW)
|
|
code = 7;
|
|
else if (result == VKEY_LTARROW)
|
|
code = 6;
|
|
else if (result == VKEY_RTARROW)
|
|
code = 8;
|
|
|
|
GotMessage = true;
|
|
mayShowCrumbIndicator();
|
|
_graphics->screenUpdate();
|
|
if (!from_crumbs(GADGETUP, code, 0, _event->updateAndGetMousePos(), curInv, curMsg, forceDraw, code, actionMode))
|
|
break;
|
|
}
|
|
}
|
|
|
|
mayShowCrumbIndicator();
|
|
_graphics->screenUpdate();
|
|
} else {
|
|
GotMessage = true;
|
|
|
|
Common::Point curPos;
|
|
curPos.x = curMsg->mouseX;
|
|
curPos.y = curMsg->mouseY;
|
|
|
|
_followingCrumbs = false;
|
|
if (!from_crumbs(curMsg->msgClass, curMsg->code, curMsg->qualifier, curPos, curInv, curMsg, forceDraw, curMsg->gadgetID, actionMode))
|
|
break;
|
|
}
|
|
}
|
|
|
|
delete _conditions;
|
|
delete _roomsFound;
|
|
|
|
if (_rooms) {
|
|
free(_rooms);
|
|
_rooms = nullptr;
|
|
}
|
|
|
|
if (_inventory) {
|
|
for (int i = 1; i <= NumInv; i++) {
|
|
if (_inventory[i].name)
|
|
free(_inventory[i].name);
|
|
|
|
if (_inventory[i].BInvName)
|
|
free(_inventory[i].BInvName);
|
|
}
|
|
|
|
free(_inventory);
|
|
}
|
|
}
|
|
|
|
bool LabEngine::from_crumbs(uint32 tmpClass, uint16 code, uint16 Qualifier, Common::Point tmpPos, uint16 &curInv, IntuiMessage *curMsg, bool &forceDraw, uint16 gadgetId, uint16 &actionMode) {
|
|
uint32 msgClass = tmpClass;
|
|
Common::Point curPos = tmpPos;
|
|
|
|
uint16 OldRoomNum, OldDirection = 0;
|
|
uint16 LastInv = MAPNUM, Old;
|
|
CloseDataPtr oldcptr, tempcptr, hcptr = NULL;
|
|
ViewData *VPtr;
|
|
bool doit;
|
|
uint16 NewDir;
|
|
|
|
|
|
_anim->_doBlack = false;
|
|
|
|
if ((msgClass == RAWKEY) && (!_graphics->_longWinInFront)) {
|
|
if (code == 13) { /* The return key */
|
|
msgClass = MOUSEBUTTONS;
|
|
Qualifier = IEQUALIFIER_LEFTBUTTON;
|
|
curPos = _event->getMousePos();
|
|
} else if (getPlatform() == Common::kPlatformWindows &&
|
|
(code == 'b' || code == 'B')) { /* Start bread crumbs */
|
|
_breadCrumbs[0]._roomNum = 0;
|
|
_numCrumbs = 0;
|
|
_droppingCrumbs = true;
|
|
mayShowCrumbIndicator();
|
|
_graphics->screenUpdate();
|
|
} else if (code == 'f' || code == 'F' ||
|
|
code == 'r' || code == 'R') { /* Follow bread crumbs */
|
|
if (_droppingCrumbs) {
|
|
if (_numCrumbs > 0) {
|
|
_followingCrumbs = true;
|
|
_followCrumbsFast = (code == 'r' || code == 'R');
|
|
_isCrumbTurning = false;
|
|
_isCrumbWaiting = false;
|
|
getTime(&_crumbSecs, &_crumbMicros);
|
|
|
|
if (_alternate) {
|
|
eatMessages();
|
|
_alternate = false;
|
|
_anim->_doBlack = true;
|
|
DoNotDrawMessage = false;
|
|
|
|
MainDisplay = true;
|
|
interfaceOn(); /* Sets the correct gadget list */
|
|
_graphics->drawPanel();
|
|
drawRoomMessage(curInv, _cptr);
|
|
_graphics->screenUpdate();
|
|
}
|
|
} else {
|
|
_breadCrumbs[0]._roomNum = 0;
|
|
_droppingCrumbs = false;
|
|
|
|
// Need to hide indicator!!!!
|
|
mayShowCrumbIndicatorOff();
|
|
_graphics->screenUpdate();
|
|
}
|
|
}
|
|
} else if ((code == 315) || (code == 'x') || (code == 'X')
|
|
|| (code == 'q') || (code == 'Q')) { /* Quit? */
|
|
DoNotDrawMessage = false;
|
|
_graphics->drawMessage("Do you want to quit? (Y/N)");
|
|
doit = false;
|
|
eatMessages();
|
|
interfaceOff();
|
|
|
|
while (1) {
|
|
_music->updateMusic(); /* Make sure we check the music at least after every message */
|
|
curMsg = getMsg();
|
|
|
|
if (curMsg == NULL) { /* Does music load and next animation frame when you've run out of messages */
|
|
_music->updateMusic();
|
|
_anim->diffNextFrame();
|
|
} else {
|
|
if (curMsg->msgClass == RAWKEY) {
|
|
if ((curMsg->code == 'Y') || (curMsg->code == 'y') || (curMsg->code == 'Q') || (curMsg->code == 'q')) {
|
|
doit = true;
|
|
break;
|
|
} else if (curMsg->code < 128) {
|
|
break;
|
|
}
|
|
} else if (curMsg->msgClass == MOUSEBUTTONS) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (doit) {
|
|
_anim->stopDiff();
|
|
return false;
|
|
} else {
|
|
forceDraw = true;
|
|
interfaceOn();
|
|
}
|
|
} else if (code == 9) { /* TAB key */
|
|
msgClass = DELTAMOVE;
|
|
} else if (code == 27) { /* ESC key */
|
|
_cptr = NULL;
|
|
}
|
|
|
|
eatMessages();
|
|
}
|
|
|
|
if (_graphics->_longWinInFront) {
|
|
if ((msgClass == RAWKEY) ||
|
|
((msgClass == MOUSEBUTTONS) &&
|
|
((IEQUALIFIER_LEFTBUTTON & Qualifier) ||
|
|
(IEQUALIFIER_RBUTTON & Qualifier)))) {
|
|
_graphics->_longWinInFront = false;
|
|
DoNotDrawMessage = false;
|
|
_graphics->drawPanel();
|
|
drawRoomMessage(curInv, _cptr);
|
|
_graphics->screenUpdate();
|
|
}
|
|
} else if ((msgClass == GADGETUP) && !_alternate) {
|
|
if (gadgetId <= 5) {
|
|
if ((actionMode == 4) && (gadgetId == 4) && (_cptr != NULL)) {
|
|
doMainView(&_cptr);
|
|
|
|
_anim->_doBlack = true;
|
|
hcptr = NULL;
|
|
_cptr = NULL;
|
|
mayShowCrumbIndicator();
|
|
_graphics->screenUpdate();
|
|
} else if (gadgetId == 5) {
|
|
eatMessages();
|
|
|
|
_alternate = true;
|
|
_anim->_doBlack = true;
|
|
DoNotDrawMessage = false;
|
|
interfaceOn(); /* Sets the correct gadget list */
|
|
|
|
MainDisplay = false;
|
|
|
|
if (LastInv && _conditions->in(LastInv)) {
|
|
curInv = LastInv;
|
|
_nextFileName = getInvName(curInv);
|
|
} else
|
|
decIncInv(&curInv, false);
|
|
|
|
_graphics->drawPanel();
|
|
drawRoomMessage(curInv, _cptr);
|
|
|
|
mayShowCrumbIndicator();
|
|
_graphics->screenUpdate();
|
|
} else {
|
|
Old = actionMode;
|
|
actionMode = gadgetId;
|
|
|
|
if (Old < 5)
|
|
perFlipGadget(Old);
|
|
|
|
perFlipGadget(actionMode);
|
|
|
|
if (gadgetId <= 4)
|
|
drawStaticMessage(kTextTakeWhat + gadgetId);
|
|
_graphics->screenUpdate();
|
|
}
|
|
} else if (gadgetId == 9) {
|
|
doUse(MAPNUM);
|
|
|
|
mayShowCrumbIndicator();
|
|
_graphics->screenUpdate();
|
|
} else if (gadgetId >= 6) { /* Arrow Gadgets */
|
|
_cptr = NULL;
|
|
hcptr = NULL;
|
|
|
|
if ((gadgetId == 6) || (gadgetId == 8)) {
|
|
if (gadgetId == 6)
|
|
drawStaticMessage(kTextTurnLeft);
|
|
else
|
|
drawStaticMessage(kTextTurnRight);
|
|
|
|
_curFileName = " ";
|
|
|
|
OldDirection = Direction;
|
|
|
|
NewDir = Direction;
|
|
processArrow(&NewDir, gadgetId - 6);
|
|
doTurn(Direction, NewDir, &_cptr);
|
|
_anim->_doBlack = true;
|
|
Direction = NewDir;
|
|
forceDraw = true;
|
|
|
|
mayShowCrumbIndicator();
|
|
_graphics->screenUpdate();
|
|
} else if (gadgetId == 7) {
|
|
OldRoomNum = _roomNum;
|
|
|
|
if (doGoForward(&_cptr)) {
|
|
if (OldRoomNum == _roomNum)
|
|
_anim->_doBlack = true;
|
|
} else {
|
|
_anim->_doBlack = true;
|
|
processArrow(&Direction, gadgetId - 6);
|
|
|
|
if (OldRoomNum != _roomNum) {
|
|
drawStaticMessage(kTextGoForward);
|
|
_roomsFound->inclElement(_roomNum); /* Potentially entered a new room */
|
|
_curFileName = " ";
|
|
forceDraw = true;
|
|
} else {
|
|
_anim->_doBlack = true;
|
|
drawStaticMessage(kTextNoPath);
|
|
}
|
|
}
|
|
|
|
if (_followingCrumbs) {
|
|
if (_isCrumbTurning) {
|
|
if (Direction == OldDirection) {
|
|
_followingCrumbs = false;
|
|
}
|
|
} else {
|
|
if (_roomNum == OldRoomNum) { // didn't get there?
|
|
_followingCrumbs = false;
|
|
}
|
|
}
|
|
} else if (_droppingCrumbs && OldRoomNum != _roomNum) {
|
|
// If in surreal maze, turn off DroppingCrumbs.
|
|
// Note: These numbers were generated by parsing the
|
|
// "Maps" file, which is why they are hard-coded. Bleh!
|
|
if (_roomNum >= 245 && _roomNum <= 280) {
|
|
_followingCrumbs = false;
|
|
_droppingCrumbs = false;
|
|
_numCrumbs = 0;
|
|
_breadCrumbs[0]._roomNum = 0;
|
|
} else {
|
|
bool intersect = false;
|
|
for (int idx = 0; idx < _numCrumbs; idx++) {
|
|
if (_breadCrumbs[idx]._roomNum == _roomNum) {
|
|
_numCrumbs = idx + 1;
|
|
_breadCrumbs[_numCrumbs]._roomNum = 0;
|
|
intersect = true;
|
|
}
|
|
}
|
|
|
|
if (!intersect) {
|
|
if (_numCrumbs == MAX_CRUMBS) {
|
|
_numCrumbs = MAX_CRUMBS - 1;
|
|
memcpy(&_breadCrumbs[0], &_breadCrumbs[1], _numCrumbs * sizeof _breadCrumbs[0]);
|
|
}
|
|
|
|
_breadCrumbs[_numCrumbs]._roomNum = _roomNum;
|
|
_breadCrumbs[_numCrumbs++]._direction = Direction;
|
|
}
|
|
}
|
|
}
|
|
|
|
mayShowCrumbIndicator();
|
|
_graphics->screenUpdate();
|
|
}
|
|
}
|
|
} else if ((msgClass == GADGETUP) && _alternate) {
|
|
_anim->_doBlack = true;
|
|
|
|
if (gadgetId == 0) {
|
|
eatMessages();
|
|
_alternate = false;
|
|
_anim->_doBlack = true;
|
|
DoNotDrawMessage = false;
|
|
|
|
MainDisplay = true;
|
|
interfaceOn(); /* Sets the correct gadget list */
|
|
_graphics->drawPanel();
|
|
drawRoomMessage(curInv, _cptr);
|
|
|
|
_graphics->screenUpdate();
|
|
}
|
|
|
|
gadgetId--;
|
|
|
|
if (gadgetId == 0) {
|
|
interfaceOff();
|
|
_anim->stopDiff();
|
|
_curFileName = " ";
|
|
|
|
doit = !saveRestoreGame();
|
|
_cptr = NULL;
|
|
|
|
MainDisplay = true;
|
|
|
|
curInv = MAPNUM;
|
|
LastInv = MAPNUM;
|
|
|
|
_nextFileName = getInvName(curInv);
|
|
|
|
_graphics->drawPanel();
|
|
|
|
if (doit) {
|
|
_graphics->drawMessage("Disk operation failed.");
|
|
setPalette(initcolors, 8);
|
|
|
|
_graphics->screenUpdate();
|
|
|
|
g_system->delayMillis(1000);
|
|
} else {
|
|
_graphics->screenUpdate();
|
|
}
|
|
} else if (gadgetId == 1) {
|
|
if (!doUse(curInv)) {
|
|
Old = actionMode;
|
|
actionMode = 5; /* Use button */
|
|
|
|
if (Old < 5)
|
|
perFlipGadget(Old);
|
|
|
|
drawStaticMessage(kTextUseOnWhat);
|
|
MainDisplay = true;
|
|
|
|
_graphics->screenUpdate();
|
|
}
|
|
} else if (gadgetId == 2) {
|
|
MainDisplay = !MainDisplay;
|
|
|
|
if ((curInv == 0) || (curInv > NumInv)) {
|
|
curInv = 1;
|
|
|
|
while ((curInv <= NumInv) && (!_conditions->in(curInv)))
|
|
curInv++;
|
|
}
|
|
|
|
if ((curInv <= NumInv) && _conditions->in(curInv) &&
|
|
_inventory[curInv].BInvName)
|
|
_nextFileName = getInvName(curInv);
|
|
|
|
_graphics->screenUpdate();
|
|
} else if (gadgetId == 3) { /* Left gadget */
|
|
decIncInv(&curInv, true);
|
|
LastInv = curInv;
|
|
DoNotDrawMessage = false;
|
|
drawRoomMessage(curInv, _cptr);
|
|
|
|
_graphics->screenUpdate();
|
|
} else if (gadgetId == 4) { /* Right gadget */
|
|
decIncInv(&curInv, false);
|
|
LastInv = curInv;
|
|
DoNotDrawMessage = false;
|
|
drawRoomMessage(curInv, _cptr);
|
|
|
|
_graphics->screenUpdate();
|
|
} else if (gadgetId == 5) { /* bread crumbs */
|
|
_breadCrumbs[0]._roomNum = 0;
|
|
_numCrumbs = 0;
|
|
_droppingCrumbs = true;
|
|
mayShowCrumbIndicator();
|
|
_graphics->screenUpdate();
|
|
} else if (gadgetId == 6) { /* follow crumbs */
|
|
if (_droppingCrumbs) {
|
|
if (_numCrumbs > 0) {
|
|
_followingCrumbs = true;
|
|
_followCrumbsFast = false;
|
|
_isCrumbTurning = false;
|
|
_isCrumbWaiting = false;
|
|
getTime(&_crumbSecs, &_crumbMicros);
|
|
|
|
eatMessages();
|
|
_alternate = false;
|
|
_anim->_doBlack = true;
|
|
DoNotDrawMessage = false;
|
|
|
|
MainDisplay = true;
|
|
interfaceOn(); /* Sets the correct gadget list */
|
|
_graphics->drawPanel();
|
|
drawRoomMessage(curInv, _cptr);
|
|
_graphics->screenUpdate();
|
|
} else {
|
|
_breadCrumbs[0]._roomNum = 0;
|
|
_droppingCrumbs = false;
|
|
|
|
// Need to hide indicator!!!!
|
|
mayShowCrumbIndicatorOff();
|
|
_graphics->screenUpdate();
|
|
}
|
|
}
|
|
}
|
|
} else if ((msgClass == MOUSEBUTTONS) && (IEQUALIFIER_LEFTBUTTON & Qualifier) && MainDisplay) {
|
|
interfaceOff();
|
|
MainDisplay = true;
|
|
|
|
doit = false;
|
|
|
|
if (_cptr) {
|
|
if ((_cptr->CloseUpType == SPECIALLOCK) && MainDisplay) /* LAB: Labyrinth specific code */
|
|
mouseCombination(curPos);
|
|
else if ((_cptr->CloseUpType == SPECIALBRICK) && MainDisplay)
|
|
mouseTile(curPos);
|
|
else
|
|
doit = true;
|
|
} else
|
|
doit = true;
|
|
|
|
|
|
if (doit) {
|
|
hcptr = NULL;
|
|
eatMessages();
|
|
|
|
if (actionMode == 0) { /* Take something. */
|
|
if (doActionRule(Common::Point(curPos.x, curPos.y), actionMode, _roomNum, &_cptr))
|
|
_curFileName = _newFileName;
|
|
else if (takeItem(curPos.x, curPos.y, &_cptr))
|
|
drawStaticMessage(kTextTakeItem);
|
|
else if (doActionRule(curPos, TAKEDEF - 1, _roomNum, &_cptr))
|
|
_curFileName = _newFileName;
|
|
else if (doActionRule(curPos, TAKE - 1, 0, &_cptr))
|
|
_curFileName = _newFileName;
|
|
else if (curPos.y < (_graphics->VGAScaleY(149) + _graphics->SVGACord(2)))
|
|
drawStaticMessage(kTextNothing);
|
|
} else if ((actionMode == 1) /* Manipulate an object */ ||
|
|
(actionMode == 2) /* Open up a "door" */ ||
|
|
(actionMode == 3)) { /* Close a "door" */
|
|
if (doActionRule(curPos, actionMode, _roomNum, &_cptr))
|
|
_curFileName = _newFileName;
|
|
else if (!doActionRule(curPos, actionMode, 0, &_cptr)) {
|
|
if (curPos.y < (_graphics->VGAScaleY(149) + _graphics->SVGACord(2)))
|
|
drawStaticMessage(kTextNothing);
|
|
}
|
|
} else if (actionMode == 4) { /* Look at closeups */
|
|
tempcptr = _cptr;
|
|
setCurClose(curPos, &tempcptr);
|
|
|
|
if (_cptr == tempcptr) {
|
|
if (curPos.y < (_graphics->VGAScaleY(149) + _graphics->SVGACord(2)))
|
|
drawStaticMessage(kTextNothing);
|
|
} else if (tempcptr->GraphicName) {
|
|
if (*(tempcptr->GraphicName)) {
|
|
_anim->_doBlack = true;
|
|
_cptr = tempcptr;
|
|
} else if (curPos.y < (_graphics->VGAScaleY(149) + _graphics->SVGACord(2)))
|
|
drawStaticMessage(kTextNothing);
|
|
} else if (curPos.y < (_graphics->VGAScaleY(149) + _graphics->SVGACord(2)))
|
|
drawStaticMessage(kTextNothing);
|
|
} else if ((actionMode == 5) &&
|
|
_conditions->in(curInv)) { /* Use an item on something else */
|
|
if (doOperateRule(curPos.x, curPos.y, curInv, &_cptr)) {
|
|
_curFileName = _newFileName;
|
|
|
|
if (!_conditions->in(curInv))
|
|
decIncInv(&curInv, false);
|
|
} else if (curPos.y < (_graphics->VGAScaleY(149) + _graphics->SVGACord(2)))
|
|
drawStaticMessage(kTextNothing);
|
|
}
|
|
}
|
|
|
|
mayShowCrumbIndicator();
|
|
_graphics->screenUpdate();
|
|
} else if (msgClass == DELTAMOVE) {
|
|
VPtr = getViewData(_roomNum, Direction);
|
|
oldcptr = VPtr->closeUps;
|
|
|
|
if (hcptr == NULL) {
|
|
tempcptr = _cptr;
|
|
setCurClose(curPos, &tempcptr);
|
|
|
|
if ((tempcptr == NULL) || (tempcptr == _cptr)) {
|
|
if (_cptr == NULL)
|
|
hcptr = oldcptr;
|
|
else
|
|
hcptr = _cptr->SubCloseUps;
|
|
} else
|
|
hcptr = tempcptr->NextCloseUp;
|
|
} else
|
|
hcptr = hcptr->NextCloseUp;
|
|
|
|
|
|
if (hcptr == NULL) {
|
|
if (_cptr == NULL)
|
|
hcptr = oldcptr;
|
|
else
|
|
hcptr = _cptr->SubCloseUps;
|
|
}
|
|
|
|
if (hcptr)
|
|
_event->setMousePos(Common::Point(_graphics->scaleX((hcptr->x1 + hcptr->x2) / 2), _graphics->scaleY((hcptr->y1 + hcptr->y2) / 2)));
|
|
} else if ((msgClass == MOUSEBUTTONS) && (IEQUALIFIER_RBUTTON & Qualifier)) {
|
|
eatMessages();
|
|
_alternate = !_alternate;
|
|
_anim->_doBlack = true;
|
|
DoNotDrawMessage = false;
|
|
MainDisplay = true;
|
|
interfaceOn(); /* Sets the correct gadget list */
|
|
|
|
if (_alternate) {
|
|
if (LastInv && _conditions->in(LastInv))
|
|
curInv = LastInv;
|
|
else
|
|
decIncInv(&curInv, false);
|
|
}
|
|
|
|
_graphics->drawPanel();
|
|
drawRoomMessage(curInv, _cptr);
|
|
|
|
mayShowCrumbIndicator();
|
|
_graphics->screenUpdate();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void LabEngine::go() {
|
|
_isHiRes = ((getFeatures() & GF_LOWRES) == 0);
|
|
|
|
if (!_graphics->setUpScreens()) {
|
|
_isHiRes = false;
|
|
_graphics->setUpScreens();
|
|
}
|
|
|
|
_event->initMouse();
|
|
|
|
_msgFont = _resource->getFont("P:AvanteG.12");
|
|
|
|
_event->mouseHide();
|
|
|
|
Intro *intro = new Intro(this);
|
|
intro->introSequence();
|
|
delete intro;
|
|
|
|
_event->mouseShow();
|
|
mainGameLoop();
|
|
|
|
if (QuitLab) { /* Won the game */
|
|
_graphics->blackAllScreen();
|
|
_graphics->readPict("P:End/L2In.1", true);
|
|
|
|
for (uint16 i = 0; i < 120; i++) {
|
|
_music->updateMusic();
|
|
waitTOF();
|
|
}
|
|
|
|
_graphics->readPict("P:End/L2In.9", true);
|
|
_graphics->readPict("P:End/Lost", true);
|
|
|
|
warning("STUB: waitForPress");
|
|
while (!1) { // 1 means ignore SDL_ProcessInput calls
|
|
_music->updateMusic();
|
|
_anim->diffNextFrame();
|
|
waitTOF();
|
|
}
|
|
}
|
|
|
|
closeFont(_msgFont);
|
|
|
|
_graphics->freePict();
|
|
|
|
freeScreens();
|
|
|
|
_music->freeMusic();
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
/* New code to allow quick(er) return navigation in game. */
|
|
/*****************************************************************************/
|
|
int LabEngine::followCrumbs() {
|
|
// NORTH, SOUTH, EAST, WEST
|
|
static int movement[4][4] = {
|
|
{ VKEY_UPARROW, VKEY_RTARROW, VKEY_RTARROW, VKEY_LTARROW },
|
|
{ VKEY_RTARROW, VKEY_UPARROW, VKEY_LTARROW, VKEY_RTARROW },
|
|
{ VKEY_LTARROW, VKEY_RTARROW, VKEY_UPARROW, VKEY_RTARROW },
|
|
{ VKEY_RTARROW, VKEY_LTARROW, VKEY_RTARROW, VKEY_UPARROW }
|
|
};
|
|
|
|
if (_isCrumbWaiting) {
|
|
uint32 Secs;
|
|
uint32 Micros;
|
|
|
|
g_lab->timeDiff(_crumbSecs, _crumbMicros, &Secs, &Micros);
|
|
|
|
if (Secs != 0 || Micros != 0)
|
|
return 0;
|
|
|
|
_isCrumbWaiting = false;
|
|
}
|
|
|
|
if (!_isCrumbTurning)
|
|
_breadCrumbs[_numCrumbs--]._roomNum = 0;
|
|
|
|
// Is the current crumb this room? If not, logic error.
|
|
if (g_lab->_roomNum != _breadCrumbs[_numCrumbs]._roomNum) {
|
|
_numCrumbs = 0;
|
|
_breadCrumbs[0]._roomNum = 0;
|
|
_droppingCrumbs = false;
|
|
_followingCrumbs = false;
|
|
return 0;
|
|
}
|
|
|
|
int exitDir;
|
|
|
|
// which direction is last crumb
|
|
if (_breadCrumbs[_numCrumbs]._direction == EAST)
|
|
exitDir = WEST;
|
|
else if (_breadCrumbs[_numCrumbs]._direction == WEST)
|
|
exitDir = EAST;
|
|
else if (_breadCrumbs[_numCrumbs]._direction == NORTH)
|
|
exitDir = SOUTH;
|
|
else
|
|
exitDir = NORTH;
|
|
|
|
int moveDir = movement[Direction][exitDir];
|
|
|
|
if (_numCrumbs == 0) {
|
|
_isCrumbTurning = false;
|
|
_breadCrumbs[0]._roomNum = 0;
|
|
_droppingCrumbs = false;
|
|
_followingCrumbs = false;
|
|
} else {
|
|
int theDelay = (_followCrumbsFast ? ONESECOND / 4 : ONESECOND);
|
|
|
|
_isCrumbTurning = (moveDir != VKEY_UPARROW);
|
|
_isCrumbWaiting = true;
|
|
|
|
g_lab->addCurTime(theDelay / ONESECOND, theDelay % ONESECOND, &_crumbSecs, &_crumbMicros);
|
|
}
|
|
|
|
return moveDir;
|
|
}
|
|
|
|
byte dropCrumbs[] = { 0x00 };
|
|
Image dropCrumbsImage(24, 24, dropCrumbs);
|
|
|
|
void LabEngine::mayShowCrumbIndicator() {
|
|
if (getPlatform() != Common::kPlatformWindows)
|
|
return;
|
|
|
|
if (_droppingCrumbs && MainDisplay) {
|
|
_event->mouseHide();
|
|
dropCrumbsImage.drawMaskImage(612, 4);
|
|
_event->mouseShow();
|
|
}
|
|
}
|
|
|
|
byte dropCrumbsOff[] = { 0x00 };
|
|
Image dropCrumbsOffImage(24, 24, dropCrumbsOff);
|
|
|
|
void LabEngine::mayShowCrumbIndicatorOff() {
|
|
if (getPlatform() != Common::kPlatformWindows)
|
|
return;
|
|
|
|
if (MainDisplay) {
|
|
_event->mouseHide();
|
|
dropCrumbsOffImage.drawMaskImage(612, 4);
|
|
_event->mouseShow();
|
|
}
|
|
}
|
|
|
|
} // End of namespace Lab
|