mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-07 18:31:37 +00:00
1471 lines
40 KiB
C++
1471 lines
40 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/stddefines.h"
|
|
#include "lab/labfun.h"
|
|
#include "lab/diff.h"
|
|
#include "lab/text.h"
|
|
#include "lab/intro.h"
|
|
#include "lab/parsefun.h"
|
|
#include "lab/interface.h"
|
|
#include "lab/mouse.h"
|
|
#include "lab/resource.h"
|
|
|
|
namespace Lab {
|
|
|
|
const char *CurFileName = " ";
|
|
|
|
bool LongWinInFront = false;
|
|
|
|
TextFont *MsgFont;
|
|
|
|
extern bool DoBlack, waitForEffect, stopsound, DoNotDrawMessage, IsHiRes, nopalchange;
|
|
|
|
/* Global parser data */
|
|
|
|
extern RoomData *_rooms;
|
|
extern InventoryData *Inventory;
|
|
extern uint16 NumInv, RoomNum, ManyRooms, HighestCondition, Direction;
|
|
CloseDataPtr CPtr;
|
|
|
|
CrumbData BreadCrumbs[MAX_CRUMBS];
|
|
uint16 NumCrumbs;
|
|
bool DroppingCrumbs;
|
|
bool FollowingCrumbs;
|
|
bool FollowCrumbsFast;
|
|
bool IsCrumbTurning;
|
|
uint32 CrumbSecs, CrumbMicros;
|
|
bool IsCrumbWaiting;
|
|
|
|
int followCrumbs();
|
|
void mayShowCrumbIndicator();
|
|
void mayShowCrumbIndicatorOff();
|
|
|
|
bool Alternate = false, ispal = false, noupdatediff = false, MainDisplay = true, QuitLab = false;
|
|
|
|
extern const char *NewFileName; /* When ProcessRoom.c decides to change the filename
|
|
of the current picture. */
|
|
|
|
#define BUFFERSIZE 850000L
|
|
|
|
/* 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 Image *MoveImages[20], *InvImages[10];
|
|
static Gadget *MoveGadgetList, *InvGadgetList;
|
|
|
|
|
|
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 control panel display. */
|
|
/******************************************************************************/
|
|
void LabEngine::drawPanel() {
|
|
_event->mouseHide();
|
|
|
|
setAPen(3); /* Clear Area */
|
|
rectFill(0, VGAScaleY(149) + SVGACord(2), VGAScaleX(319), VGAScaleY(199));
|
|
|
|
setAPen(0); /* First Line */
|
|
drawHLine(0, VGAScaleY(149) + SVGACord(2), VGAScaleX(319));
|
|
setAPen(5); /* Second Line */
|
|
drawHLine(0, VGAScaleY(149) + 1 + SVGACord(2), VGAScaleX(319));
|
|
|
|
/* Gadget Seperators */
|
|
setAPen(0);
|
|
drawHLine(0, VGAScaleY(170), VGAScaleX(319)); /* First black line to seperate buttons */
|
|
|
|
if (!Alternate) {
|
|
setAPen(4);
|
|
drawHLine(0, VGAScaleY(170) + 1, VGAScaleX(319)); /* The horizontal lines under the black one */
|
|
drawGadgetList(MoveGadgetList);
|
|
} else {
|
|
if (g_lab->getPlatform() != Common::kPlatformWindows) {
|
|
drawVLine(VGAScaleX(124), VGAScaleY(170) + 1, VGAScaleY(199)); /* Vertical Black lines */
|
|
drawVLine(VGAScaleX(194), VGAScaleY(170) + 1, VGAScaleY(199));
|
|
} else {
|
|
drawVLine(VGAScaleX(90), VGAScaleY(170) + 1, VGAScaleY(199)); /* Vertical Black lines */
|
|
drawVLine(VGAScaleX(160), VGAScaleY(170) + 1, VGAScaleY(199));
|
|
drawVLine(VGAScaleX(230), VGAScaleY(170) + 1, VGAScaleY(199));
|
|
}
|
|
|
|
setAPen(4);
|
|
drawHLine(0, VGAScaleY(170) + 1, VGAScaleX(122)); /* The horizontal lines under the black one */
|
|
drawHLine(VGAScaleX(126), VGAScaleY(170) + 1, VGAScaleX(192));
|
|
drawHLine(VGAScaleX(196), VGAScaleY(170) + 1, VGAScaleX(319));
|
|
|
|
drawVLine(VGAScaleX(1), VGAScaleY(170) + 2, VGAScaleY(198)); /* The vertical high light lines */
|
|
if (g_lab->getPlatform() != Common::kPlatformWindows) {
|
|
drawVLine(VGAScaleX(126), VGAScaleY(170) + 2, VGAScaleY(198));
|
|
drawVLine(VGAScaleX(196), VGAScaleY(170) + 2, VGAScaleY(198));
|
|
} else {
|
|
drawVLine(VGAScaleX(92), VGAScaleY(170) + 2, VGAScaleY(198));
|
|
drawVLine(VGAScaleX(162), VGAScaleY(170) + 2, VGAScaleY(198));
|
|
drawVLine(VGAScaleX(232), VGAScaleY(170) + 2, VGAScaleY(198));
|
|
}
|
|
|
|
drawGadgetList(InvGadgetList);
|
|
}
|
|
|
|
_event->mouseShow();
|
|
}
|
|
|
|
|
|
extern bool LastMessageLong;
|
|
|
|
static bool LastTooLong = false;
|
|
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
/* Draws the message for the room. */
|
|
/******************************************************************************/
|
|
void LabEngine::drawRoomMessage(uint16 CurInv, CloseDataPtr cptr) {
|
|
if (LastTooLong) {
|
|
LastTooLong = false;
|
|
return;
|
|
}
|
|
|
|
if (Alternate) {
|
|
if ((CurInv <= NumInv) && g_lab->_conditions->in(CurInv) && Inventory[CurInv].BInvName) {
|
|
if ((CurInv == LAMPNUM) && g_lab->_conditions->in(LAMPON)) /* LAB: Labyrith specific */
|
|
drawStaticMessage(kTextLampOn);
|
|
else if (Inventory[CurInv].Many > 1) {
|
|
Common::String roomMessage = Common::String(Inventory[CurInv].name) + " (" + Common::String::format("%d", Inventory[CurInv].Many) + ")";
|
|
drawMessage(roomMessage.c_str());
|
|
} else
|
|
drawMessage(Inventory[CurInv].name);
|
|
}
|
|
} else
|
|
drawDirection(cptr);
|
|
|
|
LastTooLong = LastMessageLong;
|
|
}
|
|
|
|
|
|
/******************************************************************************/
|
|
/* Sets up the Labyrinth screens, and opens up the initial windows. */
|
|
/******************************************************************************/
|
|
bool LabEngine::setUpScreens() {
|
|
byte *buffer;
|
|
byte *MovePanelBuffer, *InvPanelBuffer;
|
|
Gadget *curgad;
|
|
uint16 y;
|
|
|
|
if (!createScreen(IsHiRes))
|
|
return false;
|
|
|
|
/* Loads in the graphics for the movement control panel */
|
|
Common::File file;
|
|
file.open(translateFileName("P:Control"));
|
|
if (!file.isOpen())
|
|
warning("setUpScreens couldn't open %s", translateFileName("P:Control"));
|
|
|
|
if (file.err() || file.size() == 0)
|
|
return false;
|
|
|
|
if (!(MovePanelBuffer = (byte *)calloc(file.size(), 1)))
|
|
return false;
|
|
|
|
file.read(MovePanelBuffer, file.size());
|
|
file.close();
|
|
|
|
buffer = MovePanelBuffer;
|
|
|
|
for (uint16 i = 0; i < 20; i++)
|
|
readImage(&buffer, &(MoveImages[i]));
|
|
|
|
/* Creates the gadgets for the movement control panel */
|
|
y = VGAScaleY(173) - SVGACord(2);
|
|
|
|
if (g_lab->getPlatform() == Common::kPlatformWindows) {
|
|
MoveGadgetList = createButton(1, y, 0, 't', MoveImages[0], MoveImages[1]);
|
|
curgad = MoveGadgetList;
|
|
curgad->NextGadget = createButton(33, y, 1, 'm', MoveImages[2], MoveImages[3]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(65, y, 2, 'o', MoveImages[4], MoveImages[5]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(97, y, 3, 'c', MoveImages[6], MoveImages[7]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(129, y, 4, 'l', MoveImages[8], MoveImages[9]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(161, y, 5, 'i', MoveImages[12], MoveImages[13]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(193, y, 6, VKEY_LTARROW, MoveImages[14], MoveImages[15]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(225, y, 7, VKEY_UPARROW, MoveImages[16], MoveImages[17]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(257, y, 8, VKEY_RTARROW, MoveImages[18], MoveImages[19]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(289, y, 9, 'p', MoveImages[10], MoveImages[11]);
|
|
} else {
|
|
MoveGadgetList = createButton(1, y, 0, 0, MoveImages[0], MoveImages[1]);
|
|
curgad = MoveGadgetList;
|
|
curgad->NextGadget = createButton(33, y, 1, 0, MoveImages[2], MoveImages[3]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(65, y, 2, 0, MoveImages[4], MoveImages[5]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(97, y, 3, 0, MoveImages[6], MoveImages[7]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(129, y, 4, 0, MoveImages[8], MoveImages[9]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(161, y, 5, 0, MoveImages[12], MoveImages[13]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(193, y, 6, 0, MoveImages[14], MoveImages[15]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(225, y, 7, 0, MoveImages[16], MoveImages[17]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(257, y, 8, 0, MoveImages[18], MoveImages[19]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(289, y, 9, 0, MoveImages[10], MoveImages[11]);
|
|
}
|
|
|
|
file.open(translateFileName("P:Inv"));
|
|
if (!file.isOpen())
|
|
warning("setUpScreens couldn't open %s", translateFileName("P:Inv"));
|
|
|
|
if (file.err() || file.size() == 0)
|
|
return false;
|
|
|
|
if (!(InvPanelBuffer = (byte *)calloc(file.size(), 1)))
|
|
return false;
|
|
|
|
file.read(InvPanelBuffer, file.size());
|
|
file.close();
|
|
|
|
buffer = InvPanelBuffer;
|
|
|
|
if (g_lab->getPlatform() == Common::kPlatformWindows) {
|
|
for (uint16 imgIdx = 0; imgIdx < 10; imgIdx++)
|
|
readImage(&buffer, &(InvImages[imgIdx]));
|
|
|
|
InvGadgetList = createButton(24, y, 0, 'm', InvImages[0], InvImages[1]);
|
|
curgad = InvGadgetList;
|
|
curgad->NextGadget = createButton(56, y, 1, 'g', InvImages[2], InvImages[3]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(94, y, 2, 'u', InvImages[4], InvImages[5]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(126, y, 3, 'l', MoveImages[8], MoveImages[9]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(164, y, 4, VKEY_LTARROW, MoveImages[14], MoveImages[15]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(196, y, 5, VKEY_RTARROW, MoveImages[18], MoveImages[19]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(234, y, 6, 'b', InvImages[6], InvImages[7]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(266, y, 7, 'f', InvImages[8], InvImages[9]);
|
|
curgad = curgad->NextGadget;
|
|
} else {
|
|
for (uint16 imgIdx = 0; imgIdx < 6; imgIdx++)
|
|
readImage(&buffer, &(InvImages[imgIdx]));
|
|
|
|
InvGadgetList = createButton(58, y, 0, 0, InvImages[0], InvImages[1]);
|
|
curgad = InvGadgetList;
|
|
curgad->NextGadget = createButton(90, y, 1, 0, InvImages[2], InvImages[3]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(128, y, 2, 0, InvImages[4], InvImages[5]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(160, y, 3, 0, MoveImages[8], MoveImages[9]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(198, y, 4, 0, MoveImages[14], MoveImages[15]);
|
|
curgad = curgad->NextGadget;
|
|
curgad->NextGadget = createButton(230, y, 5, 0, MoveImages[18], MoveImages[19]);
|
|
curgad = curgad->NextGadget;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
/* Permanently flips the imagry of a gadget. */
|
|
/******************************************************************************/
|
|
void LabEngine::perFlipGadget(uint16 GadID) {
|
|
Image *Temp;
|
|
Gadget *TopGad;
|
|
|
|
TopGad = MoveGadgetList;
|
|
|
|
while (TopGad) {
|
|
if (TopGad->GadgetID == GadID) {
|
|
Temp = TopGad->Im;
|
|
TopGad->Im = TopGad->ImAlt;
|
|
TopGad->ImAlt = Temp;
|
|
|
|
if (!Alternate) {
|
|
_event->mouseHide();
|
|
drawImage(TopGad->Im, TopGad->x, TopGad->y);
|
|
_event->mouseShow();
|
|
}
|
|
|
|
return;
|
|
} else
|
|
TopGad = TopGad->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 cptr) {
|
|
if (cptr == NULL)
|
|
return false;
|
|
|
|
int monltmargin, monrtmargin, montopmargin, lutertmargin;
|
|
|
|
if (g_lab->getPlatform() != Common::kPlatformWindows) {
|
|
monltmargin = 0;
|
|
monrtmargin = 319;
|
|
montopmargin = 0;
|
|
lutertmargin = 124;
|
|
} else {
|
|
monltmargin = 2;
|
|
monrtmargin = 317;
|
|
montopmargin = 2;
|
|
lutertmargin = 128;
|
|
}
|
|
|
|
switch (cptr->CloseUpType) {
|
|
case MUSEUMMONITOR:
|
|
case LIBRARYMONITOR:
|
|
case WINDOWMONITOR:
|
|
doMonitor(cptr->GraphicName, cptr->Message, false, monltmargin, montopmargin, monrtmargin, 165);
|
|
break;
|
|
case GRAMAPHONEMONITOR:
|
|
doMonitor(cptr->GraphicName, cptr->Message, false, monltmargin, montopmargin, 171, 165);
|
|
break;
|
|
case UNICYCLEMONITOR:
|
|
doMonitor(cptr->GraphicName, cptr->Message, false, 100, montopmargin, monrtmargin, 165);
|
|
break;
|
|
case STATUEMONITOR:
|
|
doMonitor(cptr->GraphicName, cptr->Message, false, 117, montopmargin, monrtmargin, 165);
|
|
break;
|
|
case TALISMANMONITOR:
|
|
doMonitor(cptr->GraphicName, cptr->Message, false, monltmargin, montopmargin, 184, 165);
|
|
break;
|
|
case LUTEMONITOR:
|
|
doMonitor(cptr->GraphicName, cptr->Message, false, monltmargin, montopmargin, lutertmargin, 165);
|
|
break;
|
|
case CLOCKMONITOR:
|
|
doMonitor(cptr->GraphicName, cptr->Message, false, monltmargin, montopmargin, 206, 165);
|
|
break;
|
|
case TERMINALMONITOR:
|
|
doMonitor(cptr->GraphicName, cptr->Message, true, monltmargin, montopmargin, monrtmargin, 165);
|
|
break;
|
|
default:
|
|
return false;
|
|
}
|
|
|
|
CurFileName = " ";
|
|
drawPanel();
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
/* Gets the current inventory name. */
|
|
/******************************************************************************/
|
|
static const char *getInvName(uint16 CurInv) {
|
|
if (MainDisplay)
|
|
return Inventory[CurInv].BInvName;
|
|
|
|
if ((CurInv == LAMPNUM) && g_lab->_conditions->in(LAMPON))
|
|
return "P:Mines/120";
|
|
|
|
else if ((CurInv == BELTNUM) && g_lab->_conditions->in(BELTGLOW))
|
|
return "P:Future/BeltGlow";
|
|
|
|
else if (CurInv == WESTPAPERNUM) {
|
|
CurFileName = Inventory[CurInv].BInvName;
|
|
nopalchange = true;
|
|
readPict(CurFileName, false);
|
|
nopalchange = false;
|
|
doWestPaper();
|
|
}
|
|
|
|
else if (CurInv == NOTESNUM) {
|
|
CurFileName = Inventory[CurInv].BInvName;
|
|
nopalchange = true;
|
|
readPict(CurFileName, false);
|
|
nopalchange = false;
|
|
doNotes();
|
|
}
|
|
|
|
return Inventory[CurInv].BInvName;
|
|
}
|
|
|
|
static bool interfaceisoff = false;
|
|
|
|
/******************************************************************************/
|
|
/* Turns the interface off. */
|
|
/******************************************************************************/
|
|
void LabEngine::interfaceOff() {
|
|
if (!interfaceisoff) {
|
|
_event->attachGadgetList(NULL);
|
|
_event->mouseHide();
|
|
interfaceisoff = true;
|
|
}
|
|
}
|
|
|
|
/******************************************************************************/
|
|
/* Turns the interface on. */
|
|
/******************************************************************************/
|
|
void LabEngine::interfaceOn() {
|
|
if (interfaceisoff) {
|
|
interfaceisoff = false;
|
|
|
|
_event->mouseShow();
|
|
}
|
|
|
|
if (LongWinInFront)
|
|
_event->attachGadgetList(NULL);
|
|
else if (Alternate)
|
|
_event->attachGadgetList(InvGadgetList);
|
|
else
|
|
_event->attachGadgetList(MoveGadgetList);
|
|
}
|
|
|
|
static const char *Test;
|
|
|
|
/******************************************************************************/
|
|
/* 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();
|
|
stopDiff();
|
|
CurFileName = " ";
|
|
CPtr = NULL;
|
|
doMap(RoomNum);
|
|
VGASetPal(initcolors, 8);
|
|
drawMessage(NULL);
|
|
drawPanel();
|
|
} else if (CurInv == JOURNALNUM) { /* LAB: Labyrinth specific */
|
|
drawStaticMessage(kTextUseJournal);
|
|
interfaceOff();
|
|
stopDiff();
|
|
CurFileName = " ";
|
|
CPtr = NULL;
|
|
doJournal();
|
|
drawPanel();
|
|
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);
|
|
}
|
|
|
|
DoBlack = false;
|
|
waitForEffect = true;
|
|
readPict("Music:Click", true);
|
|
waitForEffect = false;
|
|
|
|
DoBlack = false;
|
|
Test = getInvName(CurInv);
|
|
} else if (CurInv == BELTNUM) { /* LAB: Labyrinth specific */
|
|
if (!_conditions->in(BELTGLOW))
|
|
_conditions->inclElement(BELTGLOW);
|
|
|
|
DoBlack = false;
|
|
Test = 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 (g_lab->_conditions->in(*CurInv) && Inventory[*CurInv].BInvName) {
|
|
Test = 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 (g_lab->_conditions->in(*CurInv) && Inventory[*CurInv].BInvName) {
|
|
Test = getInvName(*CurInv);
|
|
break;
|
|
}
|
|
|
|
if (dec)
|
|
(*CurInv)--;
|
|
else
|
|
(*CurInv)++;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/******************************************************************************/
|
|
/* The main game loop */
|
|
/******************************************************************************/
|
|
void LabEngine::mainGameLoop() {
|
|
IntuiMessage *Msg;
|
|
uint32 Class;
|
|
|
|
uint16 Qualifier, ActionMode = 4;
|
|
uint16 CurInv = MAPNUM, LastInv = MAPNUM, Old;
|
|
|
|
bool ForceDraw = false, doit, GotMessage = true;
|
|
|
|
uint16 OldRoomNum, OldDirection = 0, GadID = 0, NewDir;
|
|
|
|
CloseDataPtr OldCPtr, TempCPtr, HCPtr = NULL;
|
|
ViewData *VPtr;
|
|
|
|
VGASetPal(initcolors, 8);
|
|
|
|
CPtr = NULL;
|
|
RoomNum = 1;
|
|
Direction = NORTH;
|
|
|
|
g_resource->readRoomData("LAB:Doors");
|
|
g_resource->readInventory("LAB:Inventor");
|
|
|
|
if (!(g_lab->_conditions = new LargeSet(HighestCondition + 1)))
|
|
return;
|
|
|
|
if (!(g_lab->_roomsFound = new LargeSet(ManyRooms + 1)))
|
|
return;
|
|
|
|
g_lab->_conditions->readInitialConditions("LAB:Conditio");
|
|
|
|
LongWinInFront = false;
|
|
drawPanel();
|
|
|
|
perFlipGadget(ActionMode);
|
|
|
|
/* Set up initial picture. */
|
|
|
|
uint16 code = 0;
|
|
while (1) {
|
|
WSDL_ProcessInput(1);
|
|
|
|
if (GotMessage) {
|
|
if (QuitLab || g_engine->shouldQuit()) {
|
|
stopDiff();
|
|
break;
|
|
}
|
|
|
|
g_music->resumeBackMusic();
|
|
|
|
/* Sees what kind of close up we're in and does the appropriate stuff, if any. */
|
|
if (doCloseUp(CPtr)) {
|
|
CPtr = NULL;
|
|
|
|
mayShowCrumbIndicator();
|
|
WSDL_UpdateScreen();
|
|
}
|
|
|
|
/* Sets the current picture properly on the screen */
|
|
if (MainDisplay)
|
|
Test = getPictName(&CPtr);
|
|
|
|
if (noupdatediff) {
|
|
g_lab->_roomsFound->inclElement(RoomNum); /* Potentially entered another room */
|
|
ForceDraw = (strcmp(Test, CurFileName) != 0) || ForceDraw;
|
|
|
|
noupdatediff = false;
|
|
CurFileName = Test;
|
|
}
|
|
|
|
else if (strcmp(Test, CurFileName) != 0) {
|
|
interfaceOff();
|
|
g_lab->_roomsFound->inclElement(RoomNum); /* Potentially entered another room */
|
|
CurFileName = Test;
|
|
|
|
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
|
|
readPict(CurFileName, false);
|
|
} else
|
|
readPict(CurFileName, false);
|
|
|
|
drawRoomMessage(CurInv, CPtr);
|
|
ForceDraw = false;
|
|
|
|
mayShowCrumbIndicator();
|
|
WSDL_UpdateScreen();
|
|
|
|
if (!FollowingCrumbs)
|
|
eatMessages();
|
|
}
|
|
|
|
if (ForceDraw) {
|
|
drawRoomMessage(CurInv, CPtr);
|
|
ForceDraw = false;
|
|
WSDL_UpdateScreen();
|
|
}
|
|
}
|
|
|
|
g_music->updateMusic(); /* Make sure we check the music at least after every message */
|
|
interfaceOn();
|
|
Msg = getMsg();
|
|
|
|
Common::Point curPos;
|
|
if (Msg == NULL) { /* Does music load and next animation frame when you've run out of messages */
|
|
GotMessage = false;
|
|
g_music->checkRoomMusic();
|
|
g_music->updateMusic();
|
|
diffNextFrame();
|
|
|
|
if (FollowingCrumbs) {
|
|
int result = followCrumbs();
|
|
|
|
if (result != 0) {
|
|
curPos = WSDL_GetMousePos();
|
|
Class = GADGETUP;
|
|
Qualifier = 0;
|
|
|
|
if (result == VKEY_UPARROW)
|
|
code = GadID = 7;
|
|
else if (result == VKEY_LTARROW)
|
|
code = GadID = 6;
|
|
else if (result == VKEY_RTARROW)
|
|
code = GadID = 8;
|
|
|
|
GotMessage = true;
|
|
mayShowCrumbIndicator();
|
|
WSDL_UpdateScreen();
|
|
goto from_crumbs;
|
|
}
|
|
}
|
|
|
|
mayShowCrumbIndicator();
|
|
WSDL_UpdateScreen();
|
|
} else {
|
|
GotMessage = true;
|
|
|
|
Class = Msg->msgClass;
|
|
code = Msg->code;
|
|
Qualifier = Msg->qualifier;
|
|
curPos.x = Msg->mouseX;
|
|
curPos.y = Msg->mouseY;
|
|
GadID = Msg->gadgetID;
|
|
|
|
FollowingCrumbs = false;
|
|
|
|
from_crumbs:
|
|
DoBlack = false;
|
|
|
|
if ((Class == RAWKEY) && (!LongWinInFront)) {
|
|
if (code == 13) { /* The return key */
|
|
Class = MOUSEBUTTONS;
|
|
Qualifier = IEQUALIFIER_LEFTBUTTON;
|
|
curPos = _event->getMousePos();
|
|
} else if (g_lab->getPlatform() == Common::kPlatformWindows &&
|
|
(code == 'b' || code == 'B')) { /* Start bread crumbs */
|
|
BreadCrumbs[0].RoomNum = 0;
|
|
NumCrumbs = 0;
|
|
DroppingCrumbs = true;
|
|
mayShowCrumbIndicator();
|
|
WSDL_UpdateScreen();
|
|
} 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;
|
|
g_lab->getTime(&CrumbSecs, &CrumbMicros);
|
|
|
|
if (Alternate) {
|
|
eatMessages();
|
|
Alternate = false;
|
|
DoBlack = true;
|
|
DoNotDrawMessage = false;
|
|
|
|
MainDisplay = true;
|
|
interfaceOn(); /* Sets the correct gadget list */
|
|
drawPanel();
|
|
drawRoomMessage(CurInv, CPtr);
|
|
WSDL_UpdateScreen();
|
|
}
|
|
} else {
|
|
BreadCrumbs[0].RoomNum = 0;
|
|
DroppingCrumbs = false;
|
|
|
|
// Need to hide indicator!!!!
|
|
mayShowCrumbIndicatorOff();
|
|
WSDL_UpdateScreen();
|
|
}
|
|
}
|
|
} else if ((code == 315) || (code == 'x') || (code == 'X')
|
|
|| (code == 'q') || (code == 'Q')) { /* Quit? */
|
|
DoNotDrawMessage = false;
|
|
drawMessage("Do you want to quit? (Y/N)");
|
|
doit = false;
|
|
eatMessages();
|
|
interfaceOff();
|
|
|
|
while (1) {
|
|
g_music->updateMusic(); /* Make sure we check the music at least after every message */
|
|
Msg = getMsg();
|
|
|
|
if (Msg == NULL) { /* Does music load and next animation frame when you've run out of messages */
|
|
g_music->updateMusic();
|
|
diffNextFrame();
|
|
} else {
|
|
if (Msg->msgClass == RAWKEY) {
|
|
if ((Msg->code == 'Y') || (Msg->code == 'y') || (Msg->code == 'Q') || (Msg->code == 'q')) {
|
|
doit = true;
|
|
break;
|
|
} else if (Msg->code < 128) {
|
|
break;
|
|
}
|
|
} else if (Msg->msgClass == MOUSEBUTTONS) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (doit) {
|
|
stopDiff();
|
|
break;
|
|
} else {
|
|
ForceDraw = true;
|
|
interfaceOn();
|
|
}
|
|
} else if (code == 9) { /* TAB key */
|
|
Class = DELTAMOVE;
|
|
} else if (code == 27) { /* ESC key */
|
|
CPtr = NULL;
|
|
}
|
|
|
|
eatMessages();
|
|
}
|
|
|
|
if (LongWinInFront) {
|
|
if ((Class == RAWKEY) ||
|
|
((Class == MOUSEBUTTONS) &&
|
|
((IEQUALIFIER_LEFTBUTTON & Qualifier) ||
|
|
(IEQUALIFIER_RBUTTON & Qualifier)))) {
|
|
LongWinInFront = false;
|
|
DoNotDrawMessage = false;
|
|
drawPanel();
|
|
drawRoomMessage(CurInv, CPtr);
|
|
WSDL_UpdateScreen();
|
|
}
|
|
} else if ((Class == GADGETUP) && !Alternate) {
|
|
if (GadID <= 5) {
|
|
if ((ActionMode == 4) && (GadID == 4) && (CPtr != NULL)) {
|
|
doMainView(&CPtr);
|
|
|
|
DoBlack = true;
|
|
HCPtr = NULL;
|
|
CPtr = NULL;
|
|
mayShowCrumbIndicator();
|
|
WSDL_UpdateScreen();
|
|
} else if (GadID == 5) {
|
|
eatMessages();
|
|
|
|
Alternate = true;
|
|
DoBlack = true;
|
|
DoNotDrawMessage = false;
|
|
interfaceOn(); /* Sets the correct gadget list */
|
|
|
|
MainDisplay = false;
|
|
|
|
if (LastInv && g_lab->_conditions->in(LastInv)) {
|
|
CurInv = LastInv;
|
|
Test = getInvName(CurInv);
|
|
} else
|
|
decIncInv(&CurInv, false);
|
|
|
|
drawPanel();
|
|
drawRoomMessage(CurInv, CPtr);
|
|
|
|
mayShowCrumbIndicator();
|
|
WSDL_UpdateScreen();
|
|
} else {
|
|
Old = ActionMode;
|
|
ActionMode = GadID;
|
|
|
|
if (Old < 5)
|
|
perFlipGadget(Old);
|
|
|
|
perFlipGadget(ActionMode);
|
|
|
|
if (GadID == 0)
|
|
drawStaticMessage(kTextTakeWhat);
|
|
else if (GadID == 1)
|
|
drawStaticMessage(kTextMoveWhat);
|
|
else if (GadID == 2)
|
|
drawStaticMessage(kTextOpenWhat);
|
|
else if (GadID == 3)
|
|
drawStaticMessage(kTextCloseWhat);
|
|
else if (GadID == 4)
|
|
drawStaticMessage(kTextLookWhat);
|
|
|
|
WSDL_UpdateScreen();
|
|
}
|
|
} else if (GadID == 9) {
|
|
doUse(MAPNUM);
|
|
|
|
mayShowCrumbIndicator();
|
|
WSDL_UpdateScreen();
|
|
} else if (GadID >= 6) { /* Arrow Gadgets */
|
|
CPtr = NULL;
|
|
HCPtr = NULL;
|
|
|
|
if ((GadID == 6) || (GadID == 8)) {
|
|
if (GadID == 6)
|
|
drawStaticMessage(kTextTurnLeft);
|
|
else
|
|
drawStaticMessage(kTextTurnRight);
|
|
|
|
CurFileName = " ";
|
|
|
|
OldDirection = Direction;
|
|
|
|
NewDir = Direction;
|
|
processArrow(&NewDir, GadID - 6);
|
|
doTurn(Direction, NewDir, &CPtr);
|
|
DoBlack = true;
|
|
Direction = NewDir;
|
|
ForceDraw = true;
|
|
|
|
mayShowCrumbIndicator();
|
|
WSDL_UpdateScreen();
|
|
} else if (GadID == 7) {
|
|
OldRoomNum = RoomNum;
|
|
|
|
if (doGoForward(&CPtr)) {
|
|
if (OldRoomNum == RoomNum)
|
|
DoBlack = true;
|
|
} else {
|
|
DoBlack = true;
|
|
processArrow(&Direction, GadID - 6);
|
|
|
|
if (OldRoomNum != RoomNum) {
|
|
drawStaticMessage(kTextGoForward);
|
|
g_lab->_roomsFound->inclElement(RoomNum); /* Potentially entered a new room */
|
|
CurFileName = " ";
|
|
ForceDraw = true;
|
|
} else {
|
|
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();
|
|
WSDL_UpdateScreen();
|
|
}
|
|
}
|
|
} else if ((Class == GADGETUP) && Alternate) {
|
|
DoBlack = true;
|
|
|
|
if (GadID == 0) {
|
|
eatMessages();
|
|
Alternate = false;
|
|
DoBlack = true;
|
|
DoNotDrawMessage = false;
|
|
|
|
MainDisplay = true;
|
|
interfaceOn(); /* Sets the correct gadget list */
|
|
drawPanel();
|
|
drawRoomMessage(CurInv, CPtr);
|
|
|
|
WSDL_UpdateScreen();
|
|
}
|
|
|
|
GadID--;
|
|
|
|
if (GadID == 0) {
|
|
interfaceOff();
|
|
stopDiff();
|
|
CurFileName = " ";
|
|
|
|
doit = !saveRestoreGame();
|
|
CPtr = NULL;
|
|
|
|
MainDisplay = true;
|
|
|
|
CurInv = MAPNUM;
|
|
LastInv = MAPNUM;
|
|
|
|
Test = getInvName(CurInv);
|
|
|
|
drawPanel();
|
|
|
|
if (doit) {
|
|
drawMessage("Disk operation failed.");
|
|
VGASetPal(initcolors, 8);
|
|
|
|
WSDL_UpdateScreen();
|
|
|
|
g_system->delayMillis(1000);
|
|
} else {
|
|
WSDL_UpdateScreen();
|
|
}
|
|
} else if (GadID == 1) {
|
|
if (!doUse(CurInv)) {
|
|
Old = ActionMode;
|
|
ActionMode = 5; /* Use button */
|
|
|
|
if (Old < 5)
|
|
perFlipGadget(Old);
|
|
|
|
drawStaticMessage(kTextUseOnWhat);
|
|
MainDisplay = true;
|
|
|
|
WSDL_UpdateScreen();
|
|
}
|
|
} else if (GadID == 2) {
|
|
MainDisplay = !MainDisplay;
|
|
|
|
if ((CurInv == 0) || (CurInv > NumInv)) {
|
|
CurInv = 1;
|
|
|
|
while ((CurInv <= NumInv) && (!g_lab->_conditions->in(CurInv)))
|
|
CurInv++;
|
|
}
|
|
|
|
if ((CurInv <= NumInv) && g_lab->_conditions->in(CurInv) &&
|
|
Inventory[CurInv].BInvName)
|
|
Test = getInvName(CurInv);
|
|
|
|
WSDL_UpdateScreen();
|
|
} else if (GadID == 3) { /* Left gadget */
|
|
decIncInv(&CurInv, true);
|
|
LastInv = CurInv;
|
|
DoNotDrawMessage = false;
|
|
drawRoomMessage(CurInv, CPtr);
|
|
|
|
WSDL_UpdateScreen();
|
|
} else if (GadID == 4) { /* Right gadget */
|
|
decIncInv(&CurInv, false);
|
|
LastInv = CurInv;
|
|
DoNotDrawMessage = false;
|
|
drawRoomMessage(CurInv, CPtr);
|
|
|
|
WSDL_UpdateScreen();
|
|
} else if (GadID == 5) { /* bread crumbs */
|
|
BreadCrumbs[0].RoomNum = 0;
|
|
NumCrumbs = 0;
|
|
DroppingCrumbs = true;
|
|
mayShowCrumbIndicator();
|
|
WSDL_UpdateScreen();
|
|
} else if (GadID == 6) { /* follow crumbs */
|
|
if (DroppingCrumbs) {
|
|
if (NumCrumbs > 0) {
|
|
FollowingCrumbs = true;
|
|
FollowCrumbsFast = false;
|
|
IsCrumbTurning = false;
|
|
IsCrumbWaiting = false;
|
|
g_lab->getTime(&CrumbSecs, &CrumbMicros);
|
|
|
|
eatMessages();
|
|
Alternate = false;
|
|
DoBlack = true;
|
|
DoNotDrawMessage = false;
|
|
|
|
MainDisplay = true;
|
|
interfaceOn(); /* Sets the correct gadget list */
|
|
drawPanel();
|
|
drawRoomMessage(CurInv, CPtr);
|
|
WSDL_UpdateScreen();
|
|
} else {
|
|
BreadCrumbs[0].RoomNum = 0;
|
|
DroppingCrumbs = false;
|
|
|
|
// Need to hide indicator!!!!
|
|
mayShowCrumbIndicatorOff();
|
|
WSDL_UpdateScreen();
|
|
}
|
|
}
|
|
}
|
|
} else if ((Class == 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 < (VGAScaleY(149) + 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 < (VGAScaleY(149) + SVGACord(2)))
|
|
drawStaticMessage(kTextNothing);
|
|
}
|
|
} else if (ActionMode == 4) { /* Look at closeups */
|
|
TempCPtr = CPtr;
|
|
setCurClose(curPos, &TempCPtr);
|
|
|
|
if (CPtr == TempCPtr) {
|
|
if (curPos.y < (VGAScaleY(149) + SVGACord(2)))
|
|
drawStaticMessage(kTextNothing);
|
|
} else if (TempCPtr->GraphicName) {
|
|
if (*(TempCPtr->GraphicName)) {
|
|
DoBlack = true;
|
|
CPtr = TempCPtr;
|
|
} else if (curPos.y < (VGAScaleY(149) + SVGACord(2)))
|
|
drawStaticMessage(kTextNothing);
|
|
} else if (curPos.y < (VGAScaleY(149) + SVGACord(2)))
|
|
drawStaticMessage(kTextNothing);
|
|
} else if ((ActionMode == 5) &&
|
|
g_lab->_conditions->in(CurInv)) { /* Use an item on something else */
|
|
if (doOperateRule(curPos.x, curPos.y, CurInv, &CPtr)) {
|
|
CurFileName = NewFileName;
|
|
|
|
if (!g_lab->_conditions->in(CurInv))
|
|
decIncInv(&CurInv, false);
|
|
} else if (curPos.y < (VGAScaleY(149) + SVGACord(2)))
|
|
drawStaticMessage(kTextNothing);
|
|
}
|
|
}
|
|
|
|
mayShowCrumbIndicator();
|
|
WSDL_UpdateScreen();
|
|
} else if (Class == 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(scaleX((HCPtr->x1 + HCPtr->x2) / 2), scaleY((HCPtr->y1 + HCPtr->y2) / 2)));
|
|
} else if ((Class == MOUSEBUTTONS) && (IEQUALIFIER_RBUTTON & Qualifier)) {
|
|
eatMessages();
|
|
Alternate = !Alternate;
|
|
DoBlack = true;
|
|
DoNotDrawMessage = false;
|
|
MainDisplay = true;
|
|
interfaceOn(); /* Sets the correct gadget list */
|
|
|
|
if (Alternate) {
|
|
if (LastInv && g_lab->_conditions->in(LastInv))
|
|
CurInv = LastInv;
|
|
else
|
|
decIncInv(&CurInv, false);
|
|
}
|
|
|
|
drawPanel();
|
|
drawRoomMessage(CurInv, CPtr);
|
|
|
|
mayShowCrumbIndicator();
|
|
WSDL_UpdateScreen();
|
|
}
|
|
}
|
|
}
|
|
|
|
delete g_lab->_conditions;
|
|
delete g_lab->_roomsFound;
|
|
|
|
if (_rooms) {
|
|
free(_rooms);
|
|
_rooms = nullptr;
|
|
}
|
|
|
|
if (Inventory) {
|
|
for (code = 1; code <= NumInv; code++) {
|
|
if (Inventory[code].name)
|
|
free(Inventory[code].name);
|
|
|
|
if (Inventory[code].BInvName)
|
|
free(Inventory[code].BInvName);
|
|
}
|
|
|
|
free(Inventory);
|
|
}
|
|
}
|
|
|
|
void LabEngine::go() {
|
|
bool dointro = true;
|
|
|
|
IsHiRes = ((getFeatures() & GF_LOWRES) == 0);
|
|
|
|
#if 0
|
|
if (IsHiRes)
|
|
warning("Running in HiRes mode");
|
|
else
|
|
warning("Running in LowRes mode");
|
|
#endif
|
|
|
|
bool mem = false;
|
|
if (initBuffer(BUFFERSIZE, true)) {
|
|
mem = true;
|
|
} else {
|
|
warning("initBuffer() failed");
|
|
return;
|
|
}
|
|
|
|
if (!setUpScreens()) {
|
|
IsHiRes = false;
|
|
mem = mem && setUpScreens();
|
|
}
|
|
|
|
_event->initMouse();
|
|
|
|
mem = mem && initRoomBuffer();
|
|
|
|
if (!dointro)
|
|
g_music->initMusic();
|
|
|
|
MsgFont = g_resource->getFont("P:AvanteG.12");
|
|
|
|
_event->mouseHide();
|
|
|
|
if (dointro && mem) {
|
|
Intro intro;
|
|
intro.introSequence();
|
|
} else
|
|
DoBlack = true;
|
|
|
|
if (mem) {
|
|
_event->mouseShow();
|
|
mainGameLoop();
|
|
} else
|
|
debug("\n\nNot enough memory to start game.\n\n");
|
|
|
|
if (QuitLab) { /* Won the game */
|
|
blackAllScreen();
|
|
readPict("P:End/L2In.1", true);
|
|
|
|
for (uint16 i = 0; i < 120; i++) {
|
|
g_music->updateMusic();
|
|
waitTOF();
|
|
}
|
|
|
|
readPict("P:End/L2In.9", true);
|
|
readPict("P:End/Lost", true);
|
|
|
|
warning("STUB: waitForPress");
|
|
while (!1) { // 1 means ignore SDL_ProcessInput calls
|
|
g_music->updateMusic();
|
|
diffNextFrame();
|
|
waitTOF();
|
|
}
|
|
}
|
|
|
|
closeFont(MsgFont);
|
|
|
|
freeRoomBuffer();
|
|
freeBuffer();
|
|
|
|
g_music->freeMusic();
|
|
}
|
|
|
|
/*****************************************************************************/
|
|
/* New code to allow quick(er) return navigation in game. */
|
|
/*****************************************************************************/
|
|
int 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 (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 };
|
|
byte dropCrumbsOff[] = { 0x00 };
|
|
|
|
Image DropCrumbsImage = { 24, 24, dropCrumbs };
|
|
Image DropCrumbsOffImage = { 24, 24, dropCrumbsOff };
|
|
|
|
void LabEngine::mayShowCrumbIndicator() {
|
|
if (getPlatform() != Common::kPlatformWindows)
|
|
return;
|
|
|
|
if (DroppingCrumbs && MainDisplay) {
|
|
_event->mouseHide();
|
|
drawMaskImage(&DropCrumbsImage, 612, 4);
|
|
_event->mouseShow();
|
|
}
|
|
}
|
|
|
|
void LabEngine::mayShowCrumbIndicatorOff() {
|
|
if (getPlatform() != Common::kPlatformWindows)
|
|
return;
|
|
|
|
if (MainDisplay) {
|
|
_event->mouseHide();
|
|
drawMaskImage(&DropCrumbsOffImage, 612, 4);
|
|
_event->mouseShow();
|
|
}
|
|
}
|
|
|
|
} // End of namespace Lab
|