scummvm/engines/cge2/snail.cpp

753 lines
17 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 original Sfinx source code
* Copyright (c) 1994-1997 Janus B. Wisniewski and L.K. Avalon
*/
#include "cge2/snail.h"
#include "cge2/fileio.h"
#include "cge2/hero.h"
#include "cge2/text.h"
#include "cge2/sound.h"
#include "cge2/events.h"
namespace CGE2 {
const char *CommandHandler::_commandText[] = {
"NOP", "USE", "PAUSE", "INF", "CAVE",
"SLAVE", "FOCUS", "SETX", "SETY", "SETZ",
"ADD", "SUB", "MUL", "DIV", "IF", "FLAG",
"FLASH", "LIGHT", "CYCLE",
"CLEAR", "TALK", "MOUSE",
"MAP", "COUNT", "MIDI",
"SETDLG", "MSKDLG",
".DUMMY.",
"WAIT", "HIDE", "ROOM",
"SAY", "SOUND", "TIME", "KILL",
"RSEQ", "SEQ", "SEND", "SWAP",
"KEEP", "GIVE",
"GETPOS", "GOTO", "MOVEX", "MOVEY",
"MOVEZ", "TRANS", "PORT",
"NEXT", "NNEXT", "MTNEXT", "FTNEXT",
"RNNEXT", "RMTNEXT", "RFTNEXT",
"RMNEAR", "RMMTAKE", "RMFTAKE",
"SETREF", "BACKPT",
"WALKTO", "REACH", "COVER", "UNCOVER",
NULL };
CommandHandler::CommandHandler(CGE2Engine *vm, bool turbo)
: _turbo(turbo), _textDelay(false), _timerExpiry(0), _talkEnable(true),
_head(0), _tail(0), _commandList((Command *)malloc(sizeof(Command)* 256)),
_count(1), _vm(vm) {
}
CommandHandler::~CommandHandler() {
free(_commandList);
}
void CommandHandler::runCommand() {
if (!_turbo && _vm->_commandStat._wait) {
if (*(_vm->_commandStat._wait))
return;
else {
++_vm->_commandStat._ref[0];
warning("STUB: CommandHandler::runCommand() - Sound code missing!");
_vm->_commandStat._wait = nullptr;
}
}
uint8 tmpHead = _head;
while (_tail != tmpHead) {
Command tailCmd = _commandList[_tail];
if (!_turbo) { // only for the slower one
if (_vm->_waitRef)
break;
if (_timerExpiry) {
// Delay in progress
if (_timerExpiry > g_system->getMillis())
// Delay not yet ended
break;
// Delay is finished
_timerExpiry = 0;
} else if (_textDelay) {
if (_vm->_talk) {
_vm->snKill((Sprite *)_vm->_talk);
_vm->_talk = nullptr;
}
_textDelay = false;
}
if (_vm->_talk && tailCmd._commandType != kCmdPause)
break;
}
++_tail;
_vm->_taken = false;
Sprite *spr = nullptr;
if (tailCmd._commandType > kCmdSpr)
spr = (tailCmd._ref < 0) ? ((Sprite *)tailCmd._spritePtr) : _vm->locate(tailCmd._ref);
switch (tailCmd._commandType) {
case kCmdUse:
break;
case kCmdPause:
_timerExpiry = g_system->getMillis() + tailCmd._val * kCommandFrameDelay;
if (_vm->_talk)
_textDelay = true;
break;
case kCmdWait:
if (spr && spr->active() && (spr->_scene == _vm->_now || spr->_scene == 0)) {
_vm->_waitSeq = tailCmd._val;
_vm->_waitRef = spr->_ref;
}
break;
case kCmdHide:
_vm->snHide(spr, tailCmd._val);
break;
case kCmdSay:
_vm->snSay(spr, tailCmd._val);
break;
case kCmdInf:
if (_talkEnable)
_vm->inf(((tailCmd._val) >= 0) ? _vm->_text->getText(tailCmd._val) : (const char *)tailCmd._spritePtr);
break;
case kCmdTime:
warning("STUB: CommandHandler::runCommand() - Something missing connected to kCmdTime!");
break;
case kCmdCave:
_vm->switchCave(tailCmd._val);
break;
case kCmdMidi:
_vm->snMidi(tailCmd._val);
break;
case kCmdSetDlg:
_vm->snSetDlg(tailCmd._ref, tailCmd._val);
break;
case kCmdMskDlg:
_vm->snMskDlg(tailCmd._ref, tailCmd._val);
break;
case kCmdKill:
_vm->snKill(spr);
break;
case kCmdSeq:
_vm->snSeq(spr, tailCmd._val);
break;
case kCmdRSeq:
_vm->snRSeq(spr, tailCmd._val);
break;
case kCmdSend:
_vm->snSend(spr, tailCmd._val);
break;
case kCmdSwap:
_vm->snSwap(spr, tailCmd._val);
break;
case kCmdCover:
_vm->snCover(spr, tailCmd._val);
break;
case kCmdUncover:
_vm->snUncover(spr, (tailCmd._val >= 0) ? _vm->locate(tailCmd._val) : ((Sprite *)tailCmd._spritePtr));
break;
case kCmdFocus:
_vm->snFocus(tailCmd._val);
break;
case kCmdKeep:
_vm->snKeep(spr, tailCmd._val);
break;
case kCmdGive:
_vm->snGive(spr, tailCmd._val);
break;
case kCmdSetX:
_vm->_point[tailCmd._val]->_x = tailCmd._ref;
break;
case kCmdSetY:
_vm->_point[tailCmd._val]->_y = tailCmd._ref;
break;
case kCmdSetZ:
_vm->_point[tailCmd._val]->_z = tailCmd._ref;
break;
case kCmdAdd:
*(_vm->_point[tailCmd._ref]) = *(_vm->_point[tailCmd._ref]) + *(_vm->_point[tailCmd._val]);
break;
case kCmdSub:
*(_vm->_point[tailCmd._ref]) = *(_vm->_point[tailCmd._ref]) - *(_vm->_point[tailCmd._val]);
break;
case kCmdMul:
*(_vm->_point[tailCmd._ref]) = *(_vm->_point[tailCmd._ref]) * tailCmd._val;
break;
case kCmdDiv:
*(_vm->_point[tailCmd._ref]) = *(_vm->_point[tailCmd._ref]) / tailCmd._val;
break;
case kCmdGetPos:
if (spr)
*(_vm->_point[tailCmd._val]) = spr->_pos3D;
break;
case kCmdGoto:
_vm->snGoto(spr, tailCmd._val);
break;
case kCmdMoveX:
_vm->snMove(spr, V3D(tailCmd._val, 0, 0));
break;
case kCmdMoveY:
_vm->snMove(spr, V3D(0, tailCmd._val, 0));
break;
case kCmdMoveZ:
_vm->snMove(spr, V3D(0, 0, tailCmd._val));
break;
case kCmdSlave:
_vm->snSlave(spr, tailCmd._val);
break;
case kCmdTrans:
_vm->snTrans(spr, tailCmd._val);
break;
case kCmdPort:
_vm->snPort(spr, tailCmd._val);
break;
case kCmdNext:
break;
case kCmdIf:
break;
case kCmdTalk:
break;
case kCmdMouse:
_vm->snMouse(tailCmd._val != 0);
break;
case kCmdNNext:
_vm->snNNext(spr, kNear, tailCmd._val);
break;
case kCmdMTNext:
_vm->snNNext(spr, kMTake, tailCmd._val);
break;
case kCmdFTNext:
_vm->snNNext(spr, kFTake, tailCmd._val);
break;
case kCmdRNNext:
_vm->snRNNext(spr, tailCmd._val);
break;
case kCmdRMTNext:
_vm->snRMTNext(spr, tailCmd._val);
break;
case kCmdRFTNext:
_vm->snRFTNext(spr, tailCmd._val);
break;
case kCmdRMNear:
_vm->snRmNear(spr);
break;
case kCmdRMMTake:
_vm->snRmMTake(spr);
break;
case kCmdRMFTake:
_vm->snRmFTake(spr);
break;
case kCmdFlag:
_vm->snFlag(tailCmd._ref & 3, tailCmd._val);
break;
case kCmdSetRef:
_vm->snSetRef(spr, tailCmd._val);
break;
case kCmdBackPt:
_vm->snBackPt(spr, tailCmd._val);
break;
case kCmdFlash:
_vm->snFlash(tailCmd._val != 0);
break;
case kCmdLight:
_vm->snLight(tailCmd._val != 0);
break;
case kCmdCycle:
_vm->snCycle(tailCmd._val);
break;
case kCmdWalk:
_vm->snWalk(spr, tailCmd._val);
break;
case kCmdReach:
_vm->snReach(spr, tailCmd._val);
break;
case kCmdSound:
_vm->snSound(spr, tailCmd._val);
_count = 1;
break;
case kCmdMap:
_vm->_heroTab[tailCmd._ref & 1]->_ptr->_ignoreMap = tailCmd._val == 0;
break;
case kCmdCount:
_count = tailCmd._val;
break;
case kCmdRoom:
_vm->snRoom(spr, tailCmd._val);
break;
case kCmdDim:
_vm->snDim(spr, tailCmd._val);
break;
case kCmdExec:
warning("Unhandled command - kCmdExec");
break;
case kCmdStep:
spr->step();
break;
case kCmdGhost:
_vm->snGhost((Bitmap *)tailCmd._spritePtr);
break;
case kCmdNop: // Do nothing.
break;
default:
warning("Unhandled command");
break;
}
if (_vm->_taken && spr)
_vm->_spare->dispose(spr);
if (!_turbo)
break;
}
}
void CGE2Engine::snKill(Sprite *spr) {
if (spr) {
if (spr->_flags._kept)
releasePocket(spr);
Sprite *nx = spr->_next;
hide1(spr);
_vga->_showQ->remove(spr);
_eventManager->clearEvent(spr);
if (spr->_flags._kill)
delete spr;
else {
spr->setCave(-1);
_spare->dispose(spr);
}
if (nx && nx->_flags._slav)
snKill(nx);
}
}
void CGE2Engine::snHide(Sprite *spr, int val) {
if (spr) {
spr->_flags._hide = (val >= 0) ? (val != 0) : (!spr->_flags._hide);
if (spr->_flags._shad)
spr->_prev->_flags._hide = spr->_flags._hide;
}
}
void CGE2Engine::snMidi(int val) {
warning("STUB: CGE2Engine::snMidi()");
}
void CGE2Engine::snSetDlg(int clr, int set) {
warning("STUB: CGE2Engine::snSetDlg()");
}
void CGE2Engine::snMskDlg(int clr, int set) {
warning("STUB: CGE2Engine::snMskDlg()");
}
void CGE2Engine::snSeq(Sprite *spr, int val) {
if (spr) {
if (isHero(spr) && val == 0)
((Hero*)spr)->park();
else
spr->step(val);
}
}
void CGE2Engine::snRSeq(Sprite *spr, int val) {
if (spr)
snSeq(spr, spr->_seqPtr + val);
}
void CGE2Engine::snSend(Sprite *spr, int val) {
if (!spr)
return;
// Sending", spr->File
// from cave", spr->Cave
// to cave", val
bool was1 = (_vga->_showQ->locate(spr->_ref) != nullptr);
bool val1 = (val == 0 || val == _now);
spr->_scene = val;
releasePocket(spr);
if (val1 != was1) {
if (was1) {
// deactivating
hide1(spr);
spr->_flags._slav = false;
if (spr == _heroTab[_sex]->_ptr)
if (_heroTab[!_sex]->_ptr->_scene == _now)
switchHero(!_sex);
_spare->dispose(spr);
} else {
// activating
if (byte(spr->_ref) == 0)
_bitmapPalette = _vga->_sysPal;
_vga->_showQ->insert(spr);
if (isHero(spr)) {
V2D p = *_heroTab[spr->_ref & 1]->_posTab[val];
spr->gotoxyz(V3D(p.x, 0, p.y));
((Hero*)spr)->setCurrent();
}
_taken = false;
_bitmapPalette = NULL;
}
}
}
void CGE2Engine::snSwap(Sprite *spr, int val) {
warning("STUB: CGE2Engine::snSwap()");
}
void CGE2Engine::snCover(Sprite *spr, int val) {
warning("STUB: CGE2Engine::snCover()");
}
void CGE2Engine::snUncover(Sprite *spr, Sprite *spr2) {
warning("STUB: CGE2Engine::snUncover()");
}
void CGE2Engine::snFocus(int val) {
warning("STUB: CGE2Engine::snFocus()");
}
void CGE2Engine::snKeep(Sprite *spr, int val) {
warning("STUB: CGE2Engine::snKeep()");
}
void CGE2Engine::snGive(Sprite *spr, int val) {
warning("STUB: CGE2Engine::snGive()");
}
void CGE2Engine::snGoto(Sprite *spr, int val) {
if (spr) {
V3D eye = *_eye;
if (spr->_scene > 0)
setEye(*_eyeTab[spr->_scene]);
spr->gotoxyz(*_point[val]);
setEye(eye);
}
}
void CGE2Engine::snMove(Sprite *spr, V3D pos) {
warning("STUB: CGE2Engine::snMove()");
}
void CGE2Engine::snSlave(Sprite *spr, int val) {
warning("STUB: CGE2Engine::snSlave()");
}
void CGE2Engine::snTrans(Sprite *spr, int val) {
warning("STUB: CGE2Engine::snTrans()");
}
void CGE2Engine::snPort(Sprite *spr, int val) {
warning("STUB: CGE2Engine::snPort()");
}
void CGE2Engine::snMouse(int val) {
warning("STUB: CGE2Engine::snMouse()");
}
void CGE2Engine::snNNext(Sprite *spr, Action act, int val) {
if (spr) {
if (val > 255)
val = spr->labVal(act, val >> 8);
spr->_actionCtrl[act]._ptr = val;
}
}
void CGE2Engine::snRNNext(Sprite *spr, int val) {
warning("STUB: CGE2Engine::snRNNext()");
}
void CGE2Engine::snRMTNext(Sprite *spr, int val) {
warning("STUB: CGE2Engine::snRMTNext()");
}
void CGE2Engine::snRFTNext(Sprite *spr, int val) {
warning("STUB: CGE2Engine::snRFTNext()");
}
void CGE2Engine::snRmNear(Sprite *spr) {
warning("STUB: CGE2Engine::snRmNear()");
}
void CGE2Engine::snRmMTake(Sprite *spr) {
if (spr)
spr->_actionCtrl[kMTake]._cnt = 0;
}
void CGE2Engine::snRmFTake(Sprite *spr) {
warning("STUB: CGE2Engine::snRmFTake()");
}
void CGE2Engine::snFlag(int ref, int val) {
warning("STUB: CGE2Engine::snFlag()");
}
void CGE2Engine::snSetRef(Sprite *spr, int val) { // TODO:: Recheck this!
if (spr) {
spr->_ref = val;
_spare->update(spr);
}
}
void CGE2Engine::snBackPt(Sprite *spr, int val) {
warning("STUB: CGE2Engine::snBackPt()");
}
void CGE2Engine::snFlash(int val) {
warning("STUB: CGE2Engine::snFlash()");
}
void CGE2Engine::snLight(int val) {
warning("STUB: CGE2Engine::snLight()");
}
void CGE2Engine::snCycle(int cnt) {
warning("STUB: CGE2Engine::snCycle()");
}
void CGE2Engine::snWalk(Sprite *spr, int val) {
warning("STUB: CGE2Engine::snWalk()");
}
void CGE2Engine::snReach(Sprite *spr, int val) {
warning("STUB: CGE2Engine::snReach()");
}
void CGE2Engine::snSound(Sprite *spr, int wav) {
if (wav == -1)
_sound->stop();
else {
if (_sound->_smpinf._counter && wav < 20)
return;
if (_commandStat._wait && ((wav & 255) > 80))
return;
_commandStat._ref[1] = wav;
_commandStat._ref[0] = !_fx->exist(_commandStat._ref[1]);
_sound->play(_fx->load(_commandStat._ref[1], _commandStat._ref[0]),
(spr) ? (spr->_pos2D.x / (kScrWidth / 16)) : 8);
}
}
void CGE2Engine::snRoom(Sprite *spr, int val) {
warning("STUB: CGE2Engine::snRoom()");
}
void CGE2Engine::snDim(Sprite *spr, int val) {
if (isHero(spr)) {
if (val > 0)
++*(Hero*)spr;
if (val < 0)
--*(Hero*)spr;
}
}
void CGE2Engine::snGhost(Bitmap *bmp) {
V2D p(this, bmp->_map & 0xFFFF, bmp->_map >> 16);
bmp->hide(p.x, p.y);
delete[] bmp->_b;
bmp->_v = nullptr;
bmp->_b = nullptr;
delete bmp;
bmp = nullptr;
}
void CGE2Engine::snSay(Sprite *spr, int val) {
if (spr && spr->active() && _commandHandler->_talkEnable) {
//-- mouth animation
if (isHero(spr) && spr->seqTest(-1))
((Hero *)spr)->say();
if (_sayCap)
_text->say(_text->getText(val), spr);
if (_sayVox) {
int i = val;
if (i < 256)
i -= 100;
// TODO: These 2 lines may need some re-check with the whole sound-code!
snSound(spr, i);
_commandStat._wait = &_sound->_smpinf._counter;
}
}
}
void CGE2Engine::hide1(Sprite *spr) {
_commandHandlerTurbo->addCommand(kCmdGhost, -1, 0, spr->ghost());
}
void CommandHandler::addCommand(CommandType com, int ref, int val, void *ptr) {
if (ref == 2)
ref = 142 - _vm->_sex;
Command *headCmd = &_commandList[_head++];
headCmd->_commandType = com;
headCmd->_ref = ref;
headCmd->_val = val;
headCmd->_spritePtr = ptr;
headCmd->_cbType = kNullCB;
if (headCmd->_commandType == kCmdClear) {
clear();
}
}
void CommandHandler::addCallback(CommandType com, int ref, int val, CallbackType cbType) {
warning("STUB: CommandHandler::addCallback()");
}
void CommandHandler::insertCommand(CommandType com, int ref, int val, void *ptr) {
warning("STUB: CommandHandler::insertCommand()");
}
bool CommandHandler::idle() {
return (!_vm->_waitRef && _head == _tail);
}
void CommandHandler::reset() {
warning("STUB: CommandHandler::reset()");
}
void CommandHandler::clear() {
_tail = _head;
_vm->killText();
_timerExpiry = 0;
}
int CommandHandler::getComId(const char *com) {
int i = _vm->takeEnum(_commandText, com);
return (i < 0) ? i : i + kCmdCom0 + 1;
}
void CGE2Engine::feedSnail(Sprite *spr, Action snq, Hero *hero) {
if (!spr || !spr->active())
return;
int cnt = spr->_actionCtrl[snq]._cnt;
if (cnt) {
byte ptr = spr->_actionCtrl[snq]._ptr;
CommandHandler::Command *comtab = spr->snList(snq);
CommandHandler::Command *c = &comtab[ptr];
CommandHandler::Command *q = &comtab[cnt];
warning("STUB: CGE2Engine::feedSnail()");
// Dont bother with pockets (inventory system) for now... TODO: Implement it later!!!
/*
int pocFre = freePockets(hero->_ref & 1);
int pocReq = 0;
CommandHandler::Command *p = c;
for (; p < q && p->_commandType != kCmdNext; p++) { // scan commands
// drop from pocket?
if ((p->_commandType == kCmdSend && p->_val != _now)
|| p->_commandType == kCmdGive) {
int ref = p->_ref;
if (ref < 0)
ref = spr->_ref;
if (findActivePocket(ref) >= 0)
--pocReq;
}
// make/dispose additional room?
if (p->_commandType == kCmdRoom) {
if (p->_val == 0)
++pocReq;
else
--pocReq;
}
// put into pocket?
if (p->_commandType == kCmdKeep)
++pocReq;
// overloaded?
if (pocReq > pocFre) {
pocFul();
return;
}
}*/
while (c < q) {
if (c->_commandType == kCmdTalk) {
if ((_commandHandler->_talkEnable = (c->_val != 0)) == false)
killText();
}
if (c->_commandType == kCmdWalk || c->_commandType == kCmdReach) {
if (c->_val == -1)
c->_val = spr->_ref;
}
if (c->_commandType == kCmdNext) {
Sprite *s;
switch (c->_ref) {
case -2:
s = hero;
break;
case -1:
s = spr;
break;
default:
s = _vga->_showQ->locate(c->_ref);
break;
}
if (s) {
if (s->_actionCtrl[snq]._cnt) {
int v;
switch (c->_val) {
case -1:
v = int(c - comtab + 1);
break;
case -2:
v = int(c - comtab);
break;
case -3:
v = -1;
break;
default:
v = c->_val;
if ((v > 255) && s)
v = s->labVal(snq, v >> 8);
break;
}
if (v >= 0)
s->_actionCtrl[snq]._ptr = v;
}
}
if (s == spr)
break;
}
if (c->_commandType == kCmdIf) {
Sprite *s = (c->_ref < 0) ? spr : _vga->_showQ->locate(c->_ref);
if (s) { // sprite extsts
if (!s->seqTest(-1)) { // not parked
int v = c->_val;
if (v > 255) if (s) v = s->labVal(snq, v >> 8);
c = comtab + (v - 1);
}
}
} else
_commandHandler->addCommand(c->_commandType, c->_ref, c->_val, spr);
++c;
}
}
}
} // End of namespace CGE2.