2005-04-05 18:08:02 +00:00
|
|
|
/* ScummVM - Scumm Interpreter
|
|
|
|
* Copyright (C) 2004 Ivan Dubrov
|
2006-01-18 17:39:49 +00:00
|
|
|
* Copyright (C) 2004-2006 The ScummVM project
|
2005-04-05 18:08:02 +00:00
|
|
|
*
|
|
|
|
* 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
|
2005-04-09 19:19:54 +00:00
|
|
|
* along with this program; if not, write to the Free Software
|
2005-10-18 01:30:26 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2005-04-05 18:08:02 +00:00
|
|
|
*
|
2006-02-11 10:11:37 +00:00
|
|
|
* $URL$
|
|
|
|
* $Id$
|
2005-04-05 18:08:02 +00:00
|
|
|
*
|
|
|
|
*/
|
2006-03-29 15:59:37 +00:00
|
|
|
|
|
|
|
#include "common/stdafx.h"
|
|
|
|
#include "common/endian.h"
|
|
|
|
|
2005-04-05 15:07:40 +00:00
|
|
|
#include "gob/gob.h"
|
|
|
|
#include "gob/inter.h"
|
2007-03-20 14:51:57 +00:00
|
|
|
#include "gob/global.h"
|
2005-04-05 15:07:40 +00:00
|
|
|
#include "gob/util.h"
|
|
|
|
#include "gob/draw.h"
|
2007-03-20 14:51:57 +00:00
|
|
|
#include "gob/game.h"
|
|
|
|
#include "gob/parse.h"
|
|
|
|
#include "gob/scenery.h"
|
|
|
|
#include "gob/sound.h"
|
2005-04-05 15:07:40 +00:00
|
|
|
|
|
|
|
namespace Gob {
|
|
|
|
|
2006-01-03 23:14:39 +00:00
|
|
|
Inter::Inter(GobEngine *vm) : _vm(vm) {
|
2007-03-20 14:51:57 +00:00
|
|
|
_terminate = 0;
|
|
|
|
_break = false;
|
2006-05-11 19:43:30 +00:00
|
|
|
|
2007-03-20 14:51:57 +00:00
|
|
|
for (int i = 0; i < 8; i++) {
|
2006-05-11 19:43:30 +00:00
|
|
|
_animPalLowIndex[i] = 0;
|
|
|
|
_animPalHighIndex[i] = 0;
|
|
|
|
_animPalDir[i] = 0;
|
|
|
|
}
|
|
|
|
|
2006-01-04 01:23:20 +00:00
|
|
|
_breakFromLevel = 0;
|
|
|
|
_nestLevel = 0;
|
2006-11-20 13:03:30 +00:00
|
|
|
|
2007-03-20 14:51:57 +00:00
|
|
|
_soundEndTimeKey = 0;
|
|
|
|
_soundStopVal = 0;
|
|
|
|
|
2006-11-20 13:03:30 +00:00
|
|
|
memset(_pasteBuf, 0, 300);
|
2006-11-27 14:19:30 +00:00
|
|
|
memset(_pasteSizeBuf, 0, 300);
|
2006-11-20 13:03:30 +00:00
|
|
|
_pastePos = 0;
|
2007-03-20 14:51:57 +00:00
|
|
|
|
|
|
|
_noBusyWait = false;
|
2006-01-03 23:14:39 +00:00
|
|
|
}
|
|
|
|
|
2007-03-20 14:51:57 +00:00
|
|
|
void Inter::initControlVars(char full) {
|
|
|
|
*_nestLevel = 0;
|
|
|
|
*_breakFromLevel = -1;
|
|
|
|
|
|
|
|
*_vm->_scenery->_pCaptureCounter = 0;
|
|
|
|
|
|
|
|
_break = false;
|
|
|
|
_terminate = 0;
|
|
|
|
|
|
|
|
if (full == 1) {
|
|
|
|
for (int i = 0; i < 8; i++)
|
|
|
|
_animPalDir[i] = 0;
|
|
|
|
_soundEndTimeKey = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int16 Inter::load16() {
|
|
|
|
int16 tmp = (int16) READ_LE_UINT16(_vm->_global->_inter_execPtr);
|
2006-01-04 01:48:15 +00:00
|
|
|
_vm->_global->_inter_execPtr += 2;
|
2005-04-05 15:07:40 +00:00
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
2006-01-03 23:14:39 +00:00
|
|
|
char Inter::evalExpr(int16 *pRes) {
|
2005-04-05 15:07:40 +00:00
|
|
|
byte token;
|
|
|
|
|
2006-01-03 23:14:39 +00:00
|
|
|
_vm->_parse->printExpr(99);
|
2005-04-05 15:07:40 +00:00
|
|
|
|
2006-01-03 23:14:39 +00:00
|
|
|
_vm->_parse->parseExpr(99, &token);
|
2007-03-20 14:51:57 +00:00
|
|
|
if (!pRes)
|
2005-04-05 15:07:40 +00:00
|
|
|
return token;
|
|
|
|
|
|
|
|
switch (token) {
|
|
|
|
case 20:
|
2006-01-04 01:48:15 +00:00
|
|
|
*pRes = _vm->_global->_inter_resVal;
|
2005-04-05 15:07:40 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 22:
|
|
|
|
case 23:
|
|
|
|
*pRes = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 24:
|
|
|
|
*pRes = 1;
|
|
|
|
break;
|
|
|
|
}
|
2007-03-20 14:51:57 +00:00
|
|
|
|
2005-04-05 15:07:40 +00:00
|
|
|
return token;
|
|
|
|
}
|
|
|
|
|
2007-03-20 14:51:57 +00:00
|
|
|
bool Inter::evalBoolResult() {
|
2005-04-05 15:07:40 +00:00
|
|
|
byte token;
|
|
|
|
|
2006-01-03 23:14:39 +00:00
|
|
|
_vm->_parse->printExpr(99);
|
2005-04-05 15:07:40 +00:00
|
|
|
|
2006-01-03 23:14:39 +00:00
|
|
|
_vm->_parse->parseExpr(99, &token);
|
2007-03-20 14:51:57 +00:00
|
|
|
if ((token == 24) || ((token == 20) && _vm->_global->_inter_resVal))
|
|
|
|
return true;
|
2005-04-05 15:07:40 +00:00
|
|
|
else
|
2007-03-20 14:51:57 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Inter::renewTimeInVars() {
|
|
|
|
struct tm *t;
|
|
|
|
time_t now = time(NULL);
|
|
|
|
|
|
|
|
t = localtime(&now);
|
|
|
|
|
|
|
|
WRITE_VAR(5, 1900 + t->tm_year);
|
|
|
|
WRITE_VAR(6, t->tm_mon + 1);
|
|
|
|
WRITE_VAR(7, 0);
|
|
|
|
WRITE_VAR(8, t->tm_mday);
|
|
|
|
WRITE_VAR(9, t->tm_hour);
|
|
|
|
WRITE_VAR(10, t->tm_min);
|
|
|
|
WRITE_VAR(11, t->tm_sec);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Inter::storeMouse() {
|
|
|
|
int16 x;
|
|
|
|
int16 y;
|
|
|
|
|
|
|
|
x = _vm->_global->_inter_mouseX;
|
|
|
|
y = _vm->_global->_inter_mouseY;
|
|
|
|
_vm->_draw->adjustCoords(1, &x, &y);
|
|
|
|
|
|
|
|
WRITE_VAR(2, x);
|
|
|
|
WRITE_VAR(3, y);
|
|
|
|
WRITE_VAR(4, _vm->_game->_mouseButtons);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Inter::storeKey(int16 key) {
|
|
|
|
WRITE_VAR(12, _vm->_util->getTimeKey() - _vm->_game->_startTimeKey);
|
|
|
|
|
|
|
|
storeMouse();
|
|
|
|
WRITE_VAR(1, _vm->_snd->_playingSound);
|
|
|
|
|
|
|
|
if (key == 0x4800)
|
|
|
|
key = 0x0B;
|
|
|
|
else if (key == 0x5000)
|
|
|
|
key = 0x0A;
|
|
|
|
else if (key == 0x4D00)
|
|
|
|
key = 0x09;
|
|
|
|
else if (key == 0x4B00)
|
|
|
|
key = 0x08;
|
|
|
|
else if (key == 0x011B)
|
|
|
|
key = 0x1B;
|
|
|
|
else if (key == 0x0E08)
|
|
|
|
key = 0x19;
|
|
|
|
else if (key == 0x5300)
|
|
|
|
key = 0x1A;
|
|
|
|
else if ((key & 0xFF) != 0)
|
|
|
|
key &= 0xFF;
|
|
|
|
|
|
|
|
WRITE_VAR(0, key);
|
|
|
|
|
|
|
|
if (key != 0)
|
|
|
|
_vm->_util->clearKeyBuf();
|
2005-04-05 15:07:40 +00:00
|
|
|
}
|
|
|
|
|
2006-01-05 16:06:55 +00:00
|
|
|
void Inter::funcBlock(int16 retFlag) {
|
2007-03-20 14:51:57 +00:00
|
|
|
OpFuncParams params;
|
2006-01-05 16:06:55 +00:00
|
|
|
byte cmd;
|
|
|
|
byte cmd2;
|
2005-04-05 15:07:40 +00:00
|
|
|
|
2007-03-20 14:51:57 +00:00
|
|
|
params.retFlag = retFlag;
|
|
|
|
|
|
|
|
if (!_vm->_global->_inter_execPtr)
|
2006-01-05 16:06:55 +00:00
|
|
|
return;
|
2005-04-05 15:07:40 +00:00
|
|
|
|
2007-03-20 14:51:57 +00:00
|
|
|
_break = false;
|
2006-01-05 16:06:55 +00:00
|
|
|
_vm->_global->_inter_execPtr++;
|
2007-03-20 14:51:57 +00:00
|
|
|
params.cmdCount = *_vm->_global->_inter_execPtr++;
|
2006-01-05 16:06:55 +00:00
|
|
|
_vm->_global->_inter_execPtr += 2;
|
2005-04-05 15:07:40 +00:00
|
|
|
|
2007-03-20 14:51:57 +00:00
|
|
|
if (params.cmdCount == 0) {
|
2006-01-05 16:06:55 +00:00
|
|
|
_vm->_global->_inter_execPtr = 0;
|
|
|
|
return;
|
|
|
|
}
|
2005-04-05 15:07:40 +00:00
|
|
|
|
2007-03-20 14:51:57 +00:00
|
|
|
int startaddr = _vm->_global->_inter_execPtr - _vm->_game->_totFileData;
|
2006-03-18 13:44:57 +00:00
|
|
|
|
2007-03-20 14:51:57 +00:00
|
|
|
params.counter = 0;
|
2006-01-05 16:06:55 +00:00
|
|
|
do {
|
|
|
|
if (_terminate)
|
|
|
|
break;
|
2005-04-05 15:07:40 +00:00
|
|
|
|
2006-03-18 13:44:57 +00:00
|
|
|
// WORKAROUND:
|
|
|
|
// The EGA version of gob1 doesn't add a delay after showing
|
|
|
|
// images between levels. We manually add it here.
|
|
|
|
if ((_vm->_features & GF_GOB1) && (_vm->_features & GF_EGA)) {
|
|
|
|
int addr = _vm->_global->_inter_execPtr-_vm->_game->_totFileData;
|
|
|
|
if ((startaddr == 0x18B4 && addr == 0x1A7F && // Zombie
|
|
|
|
!strncmp(_vm->_game->_curTotFile, "avt005.tot", 10)) ||
|
|
|
|
(startaddr == 0x1299 && addr == 0x139A && // Dungeon
|
|
|
|
!strncmp(_vm->_game->_curTotFile, "avt006.tot", 10)) ||
|
|
|
|
(startaddr == 0x11C0 && addr == 0x12C9 && // Cauldron
|
|
|
|
!strncmp(_vm->_game->_curTotFile, "avt012.tot", 10)) ||
|
|
|
|
(startaddr == 0x09F2 && addr == 0x0AF3 && // Statue
|
|
|
|
!strncmp(_vm->_game->_curTotFile, "avt016.tot", 10)) ||
|
|
|
|
(startaddr == 0x0B92 && addr == 0x0C93 && // Castle
|
|
|
|
!strncmp(_vm->_game->_curTotFile, "avt019.tot", 10)) ||
|
|
|
|
(startaddr == 0x17D9 && addr == 0x18DA && // Finale
|
|
|
|
!strncmp(_vm->_game->_curTotFile, "avt022.tot", 10)))
|
|
|
|
{
|
|
|
|
_vm->_util->longDelay(5000);
|
|
|
|
}
|
2007-03-20 14:51:57 +00:00
|
|
|
} // End of workaround
|
2006-03-18 13:44:57 +00:00
|
|
|
|
2007-03-20 14:51:57 +00:00
|
|
|
cmd = (byte) *_vm->_global->_inter_execPtr;
|
2006-01-05 16:06:55 +00:00
|
|
|
if ((cmd >> 4) >= 12) {
|
|
|
|
cmd2 = 16 - (cmd >> 4);
|
2007-03-20 14:51:57 +00:00
|
|
|
cmd &= 0xF;
|
2006-01-05 16:06:55 +00:00
|
|
|
} else
|
|
|
|
cmd2 = 0;
|
2005-04-05 15:07:40 +00:00
|
|
|
|
2006-01-05 16:06:55 +00:00
|
|
|
_vm->_global->_inter_execPtr++;
|
2007-03-20 14:51:57 +00:00
|
|
|
params.counter++;
|
2005-04-05 15:07:40 +00:00
|
|
|
|
2006-01-08 20:03:20 +00:00
|
|
|
if (cmd2 == 0)
|
2006-01-05 16:06:55 +00:00
|
|
|
cmd >>= 4;
|
2005-04-05 15:07:40 +00:00
|
|
|
|
2007-03-20 14:51:57 +00:00
|
|
|
if (executeFuncOpcode(cmd2, cmd, params))
|
2006-01-05 16:06:55 +00:00
|
|
|
return;
|
2005-04-05 15:07:40 +00:00
|
|
|
|
2006-05-31 08:44:14 +00:00
|
|
|
if (_vm->_quitRequested)
|
|
|
|
break;
|
|
|
|
|
2007-03-20 14:51:57 +00:00
|
|
|
if (_break) {
|
|
|
|
if (params.retFlag != 2)
|
2006-01-05 16:06:55 +00:00
|
|
|
break;
|
2005-04-05 15:07:40 +00:00
|
|
|
|
2006-01-05 16:06:55 +00:00
|
|
|
if (*_breakFromLevel == -1)
|
2007-03-20 14:51:57 +00:00
|
|
|
_break = false;
|
2006-01-05 16:06:55 +00:00
|
|
|
break;
|
2005-04-05 15:07:40 +00:00
|
|
|
}
|
2007-03-20 14:51:57 +00:00
|
|
|
} while (params.counter != params.cmdCount);
|
2005-04-05 15:07:40 +00:00
|
|
|
|
2006-01-05 16:06:55 +00:00
|
|
|
_vm->_global->_inter_execPtr = 0;
|
|
|
|
return;
|
2005-04-05 15:07:40 +00:00
|
|
|
}
|
|
|
|
|
2006-01-05 16:06:55 +00:00
|
|
|
void Inter::callSub(int16 retFlag) {
|
|
|
|
int16 block;
|
2007-03-20 14:51:57 +00:00
|
|
|
|
|
|
|
while (!_vm->_quitRequested && _vm->_global->_inter_execPtr &&
|
|
|
|
(_vm->_global->_inter_execPtr != _vm->_game->_totFileData)) {
|
|
|
|
|
2006-01-05 16:06:55 +00:00
|
|
|
block = *_vm->_global->_inter_execPtr;
|
2007-03-20 14:51:57 +00:00
|
|
|
if (block == 1)
|
2006-01-05 16:06:55 +00:00
|
|
|
funcBlock(retFlag);
|
2007-03-20 14:51:57 +00:00
|
|
|
else if (block == 2)
|
2006-01-05 16:06:55 +00:00
|
|
|
_vm->_game->collisionsBlock();
|
2007-03-20 14:51:57 +00:00
|
|
|
else
|
|
|
|
error("Unknown block type %d in Inter::callSub()", block);
|
2006-01-05 16:06:55 +00:00
|
|
|
}
|
2005-07-19 09:10:05 +00:00
|
|
|
|
2007-03-20 14:51:57 +00:00
|
|
|
if (_vm->_global->_inter_execPtr == _vm->_game->_totFileData)
|
|
|
|
_terminate = 1;
|
2006-01-10 11:30:23 +00:00
|
|
|
}
|
|
|
|
|
2005-04-05 15:07:40 +00:00
|
|
|
} // End of namespace Gob
|