/* 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 3 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, see . * */ /* * This code is based on original Soltys source code * Copyright (c) 1994-1995 Janusz B. Wisniewski and L.K. Avalon */ #include "cge/general.h" #include "cge/sound.h" #include "cge/snail.h" #include "cge/vga13h.h" #include "cge/text.h" #include "cge/cge_main.h" #include "cge/events.h" #include "cge/walk.h" namespace CGE { const char *CommandHandler::_commandText[] = { "LABEL", "PAUSE", "WAIT", "LEVEL", "HIDE", "SAY", "INF", "TIME", "CAVE", "KILL", "RSEQ", "SEQ", "SEND", "SWAP", "KEEP", "GIVE", "IF", "GAME", "SETX0", "SETY0", "SLAVE", "SETXY", "RELX", "RELY", "RELZ", "SETX", "SETY", "SETZ", "TRANS", "PORT", "NEXT", "NNEXT", "TNEXT", "RNNEXT", "RTNEXT", "RMNEAR", "RMTAKE", "FLAG", "SETREF", "BACKPT", "FLASH", "LIGHT", "SETHB", "SETVB", "WALK", "REACH", "COVER", "UNCOVER", "CLEAR", "TALK", "MOUSE", "SOUND", "COUNT", nullptr }; CommandHandler::CommandHandler(CGEEngine *vm, bool turbo) : _turbo(turbo), _busy(false), _textDelay(false), _timerExpiry(0), _talkEnable(true), _head(0), _tail(0), _commandList((Command *)malloc(sizeof(Command) * 256)), _vm(vm) { } CommandHandler::~CommandHandler() { free(_commandList); } /** * Add a Command on the head of _commandList * @param com Command * @param ref Reference * @param val Value * @param ptr Sprite pointer */ void CommandHandler::addCommand(CommandType com, int ref, int val, void *ptr) { Command *headCmd = &_commandList[_head++]; headCmd->_commandType = com; headCmd->_ref = ref; headCmd->_val = val; headCmd->_spritePtr = ptr; headCmd->_cbType = kNullCB; if (headCmd->_commandType == kCmdClear) { _tail = _head; _vm->killText(); _timerExpiry = 0; } } /** * Add a Callback on the head of _commandList * @param com Command * @param ref Reference * @param val Value * @param CallbackType Callback type */ void CommandHandler::addCallback(CommandType com, int ref, int val, CallbackType cbType) { Command *headCmd = &_commandList[_head++]; headCmd->_commandType = com; headCmd->_ref = ref; headCmd->_val = val; headCmd->_spritePtr = nullptr; headCmd->_cbType = cbType; if (headCmd->_commandType == kCmdClear) { _tail = _head; _vm->killText(); _timerExpiry = 0; } } /** * Add a Command on the tail of _commandList * @param com Command * @param ref Reference * @param val Value * @param ptr Sprite pointer */ void CommandHandler::insertCommand(CommandType com, int ref, int val, void *ptr) { Command *tailCmd; if (_busy) { _commandList[(_tail - 1) & 0xFF] = _commandList[_tail]; tailCmd = &_commandList[_tail]; } else tailCmd = &_commandList[(_tail - 1) & 0xFF]; _tail--; tailCmd->_commandType = com; tailCmd->_ref = ref; tailCmd->_val = val; tailCmd->_spritePtr = ptr; tailCmd->_cbType = kNullCB; if (tailCmd->_commandType == kCmdClear) { _tail = _head; _vm->killText(); _timerExpiry = 0; } } void CommandHandler::runCommand() { if (_busy) return; _busy = true; uint8 tmpHead = _head; while (_tail != tmpHead) { Command *tailCmd = &_commandList[_tail]; if (!_turbo) { // only for the slower one if (_timerExpiry) { // Delay in progress if (_timerExpiry > g_system->getMillis()) // Delay not yet ended break; // Delay is finished _timerExpiry = 0; } else { if (_textDelay) { _vm->killText(); _textDelay = false; } } if (_vm->_talk && tailCmd->_commandType != kCmdPause) break; } Sprite *spr = ((tailCmd->_ref >= 0) ? _vm->locate(tailCmd->_ref) : ((Sprite *) tailCmd->_spritePtr)); switch (tailCmd->_commandType) { case kCmdLabel: break; case kCmdPause: _timerExpiry = g_system->getMillis() + tailCmd->_val * kCommandFrameDelay; if (_vm->_talk) _textDelay = true; break; case kCmdWait: if (spr) { if (spr->seqTest(tailCmd->_val) && (tailCmd->_val >= 0 || spr != _vm->_hero || _vm->_hero->_tracePtr < 0)) { _timerExpiry = g_system->getMillis() + spr->_time * kCommandFrameDelay; } else { _busy = false; return; } } break; case kCmdLevel: _vm->snLevel(spr, tailCmd->_val); break; case kCmdHide: _vm->snHide(spr, tailCmd->_val); break; case kCmdSay: if (spr && _talkEnable) { if (spr == _vm->_hero && spr->seqTest(-1)) spr->step(kSeqHTalk); _vm->_text->say(_vm->_text->getText(tailCmd->_val), spr); _vm->_sys->_funDel = kHeroFun0; } break; case kCmdInf: if (_talkEnable) { _vm->inf(_vm->_text->getText(tailCmd->_val), true); _vm->_sys->_funDel = kHeroFun0; } break; case kCmdTime: if (spr && _talkEnable) { if (spr == _vm->_hero && spr->seqTest(-1)) spr->step(kSeqHTalk); _vm->_text->sayTime(spr); } break; case kCmdCave: _vm->switchScene(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 kCmdKeep: _vm->snKeep(spr, tailCmd->_val); break; case kCmdGive: _vm->snGive(spr, tailCmd->_val); break; case kCmdGame: _vm->snGame(spr, tailCmd->_val); break; case kCmdSetX0: _vm->snSetX0(tailCmd->_ref, tailCmd->_val); break; case kCmdSetY0: _vm->snSetY0(tailCmd->_ref, tailCmd->_val); break; case kCmdSetXY: _vm->snSetXY(spr, tailCmd->_val); break; case kCmdRelX: _vm->snRelX(spr, tailCmd->_val); break; case kCmdRelY: _vm->snRelY(spr, tailCmd->_val); break; case kCmdRelZ: _vm->snRelZ(spr, tailCmd->_val); break; case kCmdSetX: _vm->snSetX(spr, tailCmd->_val); break; case kCmdSetY: _vm->snSetY(spr, tailCmd->_val); break; case kCmdSetZ: _vm->snSetZ(spr, 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: case kCmdIf: case kCmdTalk: break; case kCmdMouse: _vm->snMouse(tailCmd->_val != 0); break; case kCmdNNext: _vm->snNNext(spr, tailCmd->_val); break; case kCmdTNext: _vm->snTNext(spr, tailCmd->_val); break; case kCmdRNNext: _vm->snRNNext(spr, tailCmd->_val); break; case kCmdRTNext: _vm->snRTNext(spr, tailCmd->_val); break; case kCmdRMNear: _vm->snRmNear(spr); break; case kCmdRmTake: _vm->snRmTake(spr); break; case kCmdFlag: _vm->snFlag(tailCmd->_ref & 3, tailCmd->_val != 0); 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 kCmdSetHBarrier: _vm->snHBarrier(tailCmd->_ref, tailCmd->_val); break; case kCmdSetVBarrier: _vm->snVBarrier(tailCmd->_ref, tailCmd->_val); break; case kCmdWalk: _vm->snWalk(spr, tailCmd->_ref, tailCmd->_val); break; case kCmdReach: _vm->snReach(spr, tailCmd->_val); break; case kCmdSound: _vm->snSound(spr, tailCmd->_val); break; case kCmdCount: _vm->_sound->setRepeat(tailCmd->_val); break; case kCmdExec: switch (tailCmd->_cbType) { case kQGame: _vm->qGame(); break; case kMiniStep: _vm->miniStep(tailCmd->_val); break; case kXScene: _vm->xScene(); break; case kSoundSetVolume: _vm->sndSetVolume(); break; default: error("Unknown Callback Type in SNEXEC"); } break; case kCmdStep: spr->step(); break; case kCmdZTrim: _vm->snZTrim(spr); break; case kCmdGhost: _vm->snGhost((Bitmap *) tailCmd->_spritePtr); break; default: warning("Unhandled snc->_com in SNMouse(bool)"); break; } _tail++; if (!_turbo) break; } _busy = false; } bool CommandHandler::idle() { return (_head == _tail); } void CommandHandler::reset() { _tail = _head; } /** * Handles mini-Games logic * @param com Command * @param num mini game number */ void CGEEngine::snGame(Sprite *spr, int num) { debugC(1, kCGEDebugEngine, "CGEEngine::snGame(spr, %d)", num); switch (num) { case 1: { static Sprite *dup[3] = { nullptr, nullptr, nullptr }; int buref = 0; int Stage = 0; for (dup[0] = _vga->_showQ->first(); dup[0]; dup[0] = dup[0]->_next) { buref = dup[0]->_ref; if (buref / 1000 == 16 && buref % 100 == 6) { Stage = (buref / 100) % 10; break; } } if (dup[1] == nullptr) { dup[1] = _vga->_showQ->locate(16003); // pan dup[2] = _vga->_showQ->locate(16004); // pani } if (_game) { // continue game int i = newRandom(3), hand = (dup[0]->_shpCnt == 6); Stage++; if (hand && Stage > kDressed) ++hand; if (i >= 0 && (dup[i] == spr && newRandom(3) == 0)) { _commandHandler->addCommand(kCmdSeq, -1, 3, dup[0]); // Yes _commandHandler->addCommand(kCmdSeq, -1, 3, dup[1]); // Yes _commandHandler->addCommand(kCmdSeq, -1, 3, dup[2]); // Yes _commandHandler->addCommand(kCmdTNext, -1, 0, dup[0]); // Reset Take _commandHandler->addCommand(kCmdTNext, -1, 0, dup[1]); // Reset Take _commandHandler->addCommand(kCmdTNext, -1, 0, dup[2]); // Reset Take _commandHandler->addCommand(kCmdNNext, -1, 0, dup[0]); // Reset Near _commandHandler->addCommand(kCmdPause, -1, 72, nullptr); // Pause the game for 72/80 second _commandHandler->addCommand(kCmdSay, 1, 16009, nullptr); // Say "I win.." _commandHandler->addCommand(kCmdSay, buref, 16010, nullptr); // Say "Go Sit..." _commandHandler->addCommand(kCmdSay, 1, 16011, nullptr); // Say "I prefer not" if (hand) { _commandHandler->addCommand(kCmdSend, 16060 + hand, 16, nullptr); // Give hand _commandHandler->addCommand(kCmdSeq, buref, 4, nullptr); // Take off _commandHandler->addCommand(kCmdSeq, 16060 + hand, 1, nullptr); // start one of the Bartender animations _commandHandler->addCommand(kCmdSound, 16060 + hand, 16002, nullptr); // Play tear sound _commandHandler->addCommand(kCmdWait, 16060 + hand, 3, nullptr); // Take up _commandHandler->addCommand(kCmdSwap, buref, buref + 100, nullptr); // Open hand _commandHandler->addCommand(kCmdSeq, 16016, Stage, nullptr); // Start Belongings animation _commandHandler->addCommand(kCmdSend, 16060 + hand, -1, nullptr); // Hide hand _commandHandler->addCommand(kCmdWait, 16060 + hand, -1, nullptr); // Stop moving hand } else { _commandHandler->addCommand(kCmdSeq, buref, 4, nullptr); // Take off _commandHandler->addCommand(kCmdSound, 16060 + hand, 16002, nullptr); // Play tear sound _commandHandler->addCommand(kCmdWait, buref, -1, nullptr); // Will take off _commandHandler->addCommand(kCmdSwap, buref, buref + 100, nullptr); // Open hand _commandHandler->addCommand(kCmdSeq, 16016, Stage, nullptr); // Start Belongings animation } _commandHandler->addCommand(kCmdPause, -1, 72, nullptr); // Pause the game for 72/80 second _commandHandler->addCommand(kCmdSeq, -1, 0, dup[1]); // Get away (Him) _commandHandler->addCommand(kCmdSetXY, -1, 203 + kScrWidth * 49, dup[1]); _commandHandler->addCommand(kCmdSetZ, -1, 7, dup[1]); _commandHandler->addCommand(kCmdSeq, -1, 0, dup[2]); // Get Away (Her) _commandHandler->addCommand(kCmdSetXY, -1, 182 + kScrWidth * 62, dup[2]); _commandHandler->addCommand(kCmdSetZ, -1, 9, dup[2]); _game = false; return; } else { _commandHandler->addCommand(kCmdSeq, -1, 2, dup[0]); // reset animation sequence _commandHandler->addCommand(kCmdSeq, -1, 2, dup[1]); // reset animation sequence _commandHandler->addCommand(kCmdSeq, -1, 2, dup[2]); // reset animation sequence _commandHandler->addCommand(kCmdPause, -1, 72, nullptr); // Pause the game for 72/80 second } } _commandHandler->addCommand(kCmdWalk, 198, 134, nullptr); // Go to place _commandHandler->addCommand(kCmdWait, 1, -1, nullptr); // Stop moving _commandHandler->addCommand(kCmdCover, 1, 16101, nullptr); // Man to beat _commandHandler->addCommand(kCmdSeq, 16101, 1, nullptr); // Start Chief animation (16dupnia) _commandHandler->addCommand(kCmdWait, 16101, 5, nullptr); // wait _commandHandler->addCommand(kCmdPause, 16101, 24, nullptr); // Pause the game for 24/80 second _commandHandler->addCommand(kCmdSeq, 16040, 1, nullptr); // Start Slap animation (16plask) _commandHandler->addCommand(kCmdSound, 16101, 16001, nullptr); // Play "Slap" sound _commandHandler->addCommand(kCmdPause, 16101, 24, nullptr); // Pause the game for 24/80 second _commandHandler->addCommand(kCmdSeq, 16040, 0, nullptr); // Reset animation sequence _commandHandler->addCommand(kCmdWait, 16101, -1, nullptr); // stay _commandHandler->addCommand(kCmdUncover, 1, 16101, nullptr); // SDS if (!_game) { _commandHandler->addCommand(kCmdSay, buref, 16008, nullptr); // say "Guess!" _game = true; } } break; case 2: if (_sprTv == nullptr) { _sprTv = _vga->_showQ->locate(20700); _sprK1 = _vga->_showQ->locate(20701); _sprK2 = _vga->_showQ->locate(20702); _sprK3 = _vga->_showQ->locate(20703); } if (!_game) { // init _commandHandler->addCommand(kCmdGame, 20002, 2, nullptr); _game = true; break; } // cont _sprK1->step(newRandom(6)); _sprK2->step(newRandom(6)); _sprK3->step(newRandom(6)); // check the ALT key as it's the solution of the puzzle // the test has been restricted to some specific OSes // in order to avoid some obvious issues (like Android, iOS, NDS, N64...) // Not perfect, but at least better than nothing. #if defined(WIN32) || defined(UNIX) || defined(MACOSX) if (spr->_ref == 1 && _keyboard->_keyAlt) { #else if (spr->_ref == 1 && _gameCase2Cpt > 1) { #endif _sprK1->step(5); _sprK2->step(5); _sprK3->step(5); } _commandHandler->addCommand(kCmdSetZ, 20700, 0, nullptr); { bool hit = (_sprK1->_seqPtr + _sprK2->_seqPtr + _sprK3->_seqPtr == 15); if (hit) { if (spr->_ref == 1) { _commandHandler->addCommand(kCmdSay, 1, 20003, nullptr); // hurray! _commandHandler->addCommand(kCmdSeq, 20011, 2, nullptr); // Camera away _commandHandler->addCommand(kCmdSend, 20701, -1, nullptr); // move dice1 to scene -1 _commandHandler->addCommand(kCmdSend, 20702, -1, nullptr); // move dice2 to scene -1 _commandHandler->addCommand(kCmdSend, 20703, -1, nullptr); // move dice3 to scene -1 _commandHandler->addCommand(kCmdSend, 20700, -1, nullptr); // move TV to scene -1 _commandHandler->addCommand(kCmdKeep, 20007, 0, nullptr); // to pocket _commandHandler->addCommand(kCmdSend, 20006, 20, nullptr); // Move Coin to scene 20 _commandHandler->addCommand(kCmdSound, 20006, 20002, nullptr); // Play Coin sound _commandHandler->addCommand(kCmdSay, 20002, 20004, nullptr); // Say "Luck guy..." _commandHandler->addCommand(kCmdSend, 20010, 20, nullptr); // Move Paper to scene 20 _commandHandler->addCommand(kCmdSound, 20010, 20003, nullptr); // Play "ksh" sound! (fx20003.wav) _commandHandler->addCommand(kCmdSay, 20001, 20005, nullptr); // Say "Congratulations" _game = false; return; } else { _sprK3->step(newRandom(5)); } } } if (_gameCase2Cpt < 100) { switch (_gameCase2Cpt) { case 15: // Give hint about ALTered dice _commandHandler->addCommand(kCmdSay, 20003, 20021, nullptr); break; case 30: case 45: case 60: case 75: // Tell to use ALT key _commandHandler->addCommand(kCmdSay, 20003, 20022, nullptr); break; default: break; } _gameCase2Cpt++; } switch (spr->_ref) { case 1: _commandHandler->addCommand(kCmdSay, 20001, 20011, nullptr); // Say "It'a my turn" _commandHandler->addCommand(kCmdSeq, 20001, 1, nullptr); // Throw dice _commandHandler->addCommand(kCmdWait, 20001, 1, nullptr); // wait _commandHandler->addCommand(kCmdSetZ, 20700, 2, nullptr); // hide dice _commandHandler->addCommand(kCmdHide, 20007, 1, nullptr); // hide dice _commandHandler->addCommand(kCmdWait, 20001, 16, nullptr); // wait _commandHandler->addCommand(kCmdSeq, 20007, 1, nullptr); // Start dice animation (20kosci) _commandHandler->addCommand(kCmdHide, 20007, 0, nullptr); // unhide _commandHandler->addCommand(kCmdSound, 20007, 20001, nullptr); // Play Dice sound _commandHandler->addCommand(kCmdWait, 20007, -1, nullptr); // the end _commandHandler->addCommand(kCmdGame, 20001, 2, nullptr); // again! break; case 20001: _commandHandler->addCommand(kCmdSay, 20002, 20012, nullptr); // Say "Now it's mine" _commandHandler->addCommand(kCmdSeq, 20002, 1, nullptr); // Throw dice _commandHandler->addCommand(kCmdWait, 20002, 3, nullptr); // wait _commandHandler->addCommand(kCmdSetZ, 20700, 2, nullptr); // hide dice _commandHandler->addCommand(kCmdHide, 20007, 1, nullptr); // hide dice _commandHandler->addCommand(kCmdWait, 20002, 10, nullptr); // wait _commandHandler->addCommand(kCmdSeq, 20007, 2, nullptr); // Start dice animation (20kosci) _commandHandler->addCommand(kCmdHide, 20007, 0, nullptr); // unhide _commandHandler->addCommand(kCmdSound, 20007, 20001, nullptr); // Play Dice sound _commandHandler->addCommand(kCmdWait, 20007, -1, nullptr); // the end _commandHandler->addCommand(kCmdGame, 20002, 2, nullptr); // again! break; case 20002: _commandHandler->addCommand(kCmdSay, 20002, 20010, nullptr); // "Roll the bones!" _commandHandler->addCommand(kCmdWalk, 20005, -1, nullptr); // Walk to table _commandHandler->addCommand(kCmdWait, 1, -1, nullptr); // Wait _commandHandler->addCommand(kCmdCover, 1, 20101, nullptr); // grasol ?? _commandHandler->addCommand(kCmdSeq, 20101, 1, nullptr); // Start Chief animation (20solgra) _commandHandler->addCommand(kCmdWait, 20101, 5, nullptr); // Wait _commandHandler->addCommand(kCmdSetZ, 20700, 2, nullptr); // Hide dice _commandHandler->addCommand(kCmdHide, 20007, 1, nullptr); // Hide dice _commandHandler->addCommand(kCmdWait, 20101, 15, nullptr); // wait _commandHandler->addCommand(kCmdSeq, 20007, 1, nullptr); // Start dice animation (20kosci) _commandHandler->addCommand(kCmdHide, 20007, 0, nullptr); // Unhide _commandHandler->addCommand(kCmdSound, 20007, 20001, nullptr); // Play Dice sound _commandHandler->addCommand(kCmdWait, 20101, -1, nullptr); // the end _commandHandler->addCommand(kCmdUncover, 1, 20101, nullptr); // SDS ?? _commandHandler->addCommand(kCmdGame, 1, 2, nullptr); // again! break; default: break; } default: break; } } void CGEEngine::expandSprite(Sprite *spr) { debugC(5, kCGEDebugEngine, "CGEEngine::expandSprite(spr)"); if (spr) _vga->_showQ->insert(_vga->_spareQ->remove(spr)); } void CGEEngine::contractSprite(Sprite *spr) { debugC(1, kCGEDebugEngine, "CGEEngine::contractSprite(spr)"); if (spr) _vga->_spareQ->append(_vga->_showQ->remove(spr)); } /** * Check if an item is in the inventory, and returns its position * @param spr Sprite pointer * @return -1 if not found, else index. */ int CGEEngine::findPocket(Sprite *spr) { debugC(1, kCGEDebugEngine, "CGEEngine::findPocket(spr)"); for (int i = 0; i < kPocketNX; i++) if (_pocket[i] == spr) return i; return -1; } /** * Check if an item is in the inventory, and returns its position * @param Inventory slot number Sprite pointer */ void CGEEngine::selectPocket(int n) { debugC(1, kCGEDebugEngine, "CGEEngine::selectPocket(%d)", n); if (n < 0 || (_pocLight->_seqPtr && _pocPtr == n)) { // If no slot specified, or another slot already selected // stop the blinking animation _pocLight->step(0); n = findPocket(nullptr); if (n >= 0) _pocPtr = n; } else { // If slot specified, check if the slot if used. // Is so, start the blinking animation if (_pocket[n] != nullptr) { _pocPtr = n; _pocLight->step(1); } } _pocLight->gotoxy(kPocketX + _pocPtr * kPocketDX + kPocketSX, kPocketY + kPocketSY); } /** * Logic used when all the inventory slots are full and the user tries to pick * another object. * @param Inventory slot number Sprite pointer */ void CGEEngine::pocFul() { debugC(1, kCGEDebugEngine, "CGEEngine::pocFul()"); if (!_hero) error("pocFul - Unexpected null _hero"); _hero->park(); _commandHandler->addCommand(kCmdWait, -1, -1, _hero); _commandHandler->addCommand(kCmdSeq, -1, kSeqPocketFull, _hero); _commandHandler->addCommand(kCmdSound, -1, 2, _hero); // Play the 'hum-hum" sound (fx00002) _commandHandler->addCommand(kCmdWait, -1, -1, _hero); _commandHandler->addCommand(kCmdSay, 1, kPocketFull, _hero); } void CGEEngine::hide1(Sprite *spr) { debugC(1, kCGEDebugEngine, "CGEEngine::hide1(spr)"); _commandHandlerTurbo->addCommand(kCmdGhost, -1, 0, spr->ghost()); } void CGEEngine::snGhost(Bitmap *bmp) { debugC(1, kCGEDebugEngine, "CGEEngine::snGhost(bmp)"); bmp->hide(bmp->_map & 0xFFFF, bmp->_map >> 16); bmp->_m = nullptr; bmp->_map = 0; delete bmp; } void CGEEngine::feedSnail(Sprite *spr, SnList snq) { debugC(1, kCGEDebugEngine, "CGEEngine::feedSnail(spr, snq)"); if (!spr || !spr->active()) return; uint8 ptr = (snq == kTake) ? spr->_takePtr : spr->_nearPtr; if (ptr == kNoPtr) return; CommandHandler::Command *comtab = spr->snList(snq); CommandHandler::Command *c = comtab + ptr; if (findPocket(nullptr) < 0) { // no empty pockets? CommandHandler::Command *p; for (p = c; p->_commandType != kCmdNext; p++) { // find KEEP command if (p->_commandType == kCmdKeep) { pocFul(); return; } if (p->_spritePtr) break; } } while (true) { if (c->_commandType == kCmdTalk) { if ((_commandHandler->_talkEnable = (c->_val != 0)) == false) killText(); } if (c->_commandType == kCmdNext) { Sprite *s = (c->_ref < 0) ? spr : locate(c->_ref); if (s) { uint8 *idx = (snq == kTake) ? &s->_takePtr : &s->_nearPtr; if (*idx != kNoPtr) { int v; switch (c->_val) { case -1 : v = c - comtab + 1; break; case -2 : v = c - comtab; break; case -3 : v = -1; break; default : v = c->_val; break; } if (v >= 0) *idx = v; } } if (s == spr) break; } if (c->_commandType == kCmdIf) { Sprite *s = (c->_ref < 0) ? spr : locate(c->_ref); if (s) { // sprite extsts if (! s->seqTest(-1)) c = comtab + c->_val; // not parked else ++c; } else ++c; } else { _commandHandler->addCommand(c->_commandType, c->_ref, c->_val, spr); if (c->_spritePtr) break; else c++; } } } void CGEEngine::snNNext(Sprite *spr, int p) { debugC(1, kCGEDebugEngine, "CGEEngine::snNNext(spr, %d)", p); if (spr) if (spr->_nearPtr != kNoPtr) spr->_nearPtr = p; } void CGEEngine::snTNext(Sprite *spr, int p) { debugC(1, kCGEDebugEngine, "CGEEngine::snTNext(spr, %d)", p); if (spr) if (spr->_takePtr != kNoPtr) spr->_takePtr = p; } void CGEEngine::snRNNext(Sprite *spr, int p) { debugC(1, kCGEDebugEngine, "CGEEngine::snRNNext(spr, %d)", p); if (spr) if (spr->_nearPtr != kNoPtr) spr->_nearPtr += p; } void CGEEngine::snRTNext(Sprite *spr, int p) { debugC(1, kCGEDebugEngine, "CGEEngine::snRTNext(spr, %d)", p); if (spr) if (spr->_takePtr != kNoPtr) spr->_takePtr += p; } void CGEEngine::snZTrim(Sprite *spr) { debugC(4, kCGEDebugEngine, "CGEEngine::snZTrim(spr)"); if (!spr || !spr->active()) return; Sprite *s = (spr->_flags._shad) ? spr->_prev : nullptr; _vga->_showQ->insert(_vga->_showQ->remove(spr)); if (s) { s->_z = spr->_z; _vga->_showQ->insert(_vga->_showQ->remove(s), spr); } } void CGEEngine::snHide(Sprite *spr, int val) { debugC(1, kCGEDebugEngine, "CGEEngine::snHide(spr, %d)", 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 CGEEngine::snRmNear(Sprite *spr) { debugC(1, kCGEDebugEngine, "CGEEngine::snRmNear(spr)"); if (spr) spr->_nearPtr = kNoPtr; } void CGEEngine::snRmTake(Sprite *spr) { debugC(1, kCGEDebugEngine, "CGEEngine::snRmTake(spr)"); if (spr) spr->_takePtr = kNoPtr; } void CGEEngine::snSeq(Sprite *spr, int val) { debugC(1, kCGEDebugEngine, "CGEEngine::snSeq(spr, %d)", val); if (spr) { if (spr == _hero && val == 0) _hero->park(); else spr->step(val); } } void CGEEngine::snRSeq(Sprite *spr, int val) { debugC(1, kCGEDebugEngine, "CGEEngine::snRSeq(spr, %d)", val); if (spr) snSeq(spr, spr->_seqPtr + val); } void CGEEngine::snSend(Sprite *spr, int val) { debugC(1, kCGEDebugEngine, "CGEEngine::snSend(spr, %d)", val); if (!spr) return; int was = spr->_scene; bool was1 = (was == 0 || was == _now); bool val1 = (val == 0 || val == _now); spr->_scene = val; if (val1 != was1) { if (was1) { if (spr->_flags._kept) { int n = findPocket(spr); if (n >= 0) _pocket[n] = nullptr; } hide1(spr); contractSprite(spr); spr->_flags._slav = false; } else { if (spr->_ref % 1000 == 0) _bitmapPalette = _vga->_sysPal; if (spr->_flags._back) spr->backShow(true); else expandSprite(spr); _bitmapPalette = nullptr; } } } void CGEEngine::snSwap(Sprite *spr, int xref) { debugC(1, kCGEDebugEngine, "CGEEngine::snSwap(spr, %d)", xref); Sprite *xspr = locate(xref); if (!spr || !xspr) return; int was = spr->_scene; int xwas = xspr->_scene; bool was1 = (was == 0 || was == _now); bool xwas1 = (xwas == 0 || xwas == _now); SWAP(spr->_scene, xspr->_scene); SWAP(spr->_x, xspr->_x); SWAP(spr->_y, xspr->_y); SWAP(spr->_z, xspr->_z); if (spr->_flags._kept) { int n = findPocket(spr); if (n >= 0) _pocket[n] = xspr; xspr->_flags._kept = true; xspr->_flags._port = false; } if (xwas1 != was1) { if (was1) { hide1(spr); contractSprite(spr); } else expandSprite(spr); if (xwas1) { hide1(xspr); contractSprite(xspr); } else expandSprite(xspr); } } void CGEEngine::snCover(Sprite *spr, int xref) { debugC(1, kCGEDebugEngine, "CGEEngine::snCover(spr, %d)", xref); Sprite *xspr = locate(xref); if (!spr || !xspr) return; spr->_flags._hide = true; xspr->_z = spr->_z; xspr->_scene = spr->_scene; xspr->gotoxy(spr->_x, spr->_y); expandSprite(xspr); if ((xspr->_flags._shad = spr->_flags._shad) == 1) { _vga->_showQ->insert(_vga->_showQ->remove(spr->_prev), xspr); spr->_flags._shad = false; } feedSnail(xspr, kNear); } void CGEEngine::snUncover(Sprite *spr, Sprite *xspr) { debugC(1, kCGEDebugEngine, "CGEEngine::snUncover(spr, xspr)"); if (!spr || !xspr) return; spr->_flags._hide = false; spr->_scene = xspr->_scene; spr->gotoxy(xspr->_x, xspr->_y); if ((spr->_flags._shad = xspr->_flags._shad) == 1) { _vga->_showQ->insert(_vga->_showQ->remove(xspr->_prev), spr); xspr->_flags._shad = false; } spr->_z = xspr->_z; snSend(xspr, -1); if (spr->_time == 0) spr->_time++; } void CGEEngine::snSetX0(int scene, int x0) { debugC(1, kCGEDebugEngine, "CGEEngine::snSetX0(%d, %d)", scene, x0); _heroXY[scene - 1].x = x0; } void CGEEngine::snSetY0(int scene, int y0) { debugC(1, kCGEDebugEngine, "CGEEngine::snSetY0(%d, %d)", scene, y0); _heroXY[scene - 1].y = y0; } void CGEEngine::snSetXY(Sprite *spr, uint16 xy) { debugC(1, kCGEDebugEngine, "CGEEngine::snSetXY(spr, %d)", xy); if (spr) spr->gotoxy(xy % kScrWidth, xy / kScrWidth); } void CGEEngine::snRelX(Sprite *spr, int x) { debugC(1, kCGEDebugEngine, "CGEEngine::snRelX(spr, %d)", x); if (spr && _hero) spr->gotoxy(_hero->_x + x, spr->_y); } void CGEEngine::snRelY(Sprite *spr, int y) { debugC(1, kCGEDebugEngine, "CGEEngine::snRelY(spr, %d)", y); if (spr && _hero) spr->gotoxy(spr->_x, _hero->_y + y); } void CGEEngine::snRelZ(Sprite *spr, int z) { debugC(1, kCGEDebugEngine, "CGEEngine::snRelZ(spr, %d)", z); if (spr && _hero) { spr->_z = _hero->_z + z; snZTrim(spr); } } void CGEEngine::snSetX(Sprite *spr, int x) { debugC(1, kCGEDebugEngine, "CGEEngine::snSetX(spr, %d)", x); if (spr) spr->gotoxy(x, spr->_y); } void CGEEngine::snSetY(Sprite *spr, int y) { debugC(1, kCGEDebugEngine, "CGEEngine::snSetY(spr, %d)", y); if (spr) spr->gotoxy(spr->_x, y); } void CGEEngine::snSetZ(Sprite *spr, int z) { debugC(1, kCGEDebugEngine, "CGEEngine::snSetZ(spr, %d)", z); if (spr) { spr->_z = z; snZTrim(spr); } } void CGEEngine::snSlave(Sprite *spr, int ref) { debugC(1, kCGEDebugEngine, "CGEEngine::snSlave(spr, %d)", ref); Sprite *slv = locate(ref); if (spr && slv) { if (spr->active()) { snSend(slv, spr->_scene); slv->_flags._slav = true; slv->_z = spr->_z; _vga->_showQ->insert(_vga->_showQ->remove(slv), spr->_next); } } } void CGEEngine::snTrans(Sprite *spr, int trans) { debugC(1, kCGEDebugEngine, "CGEEngine::snTrans(spr, %d)", trans); if (spr) spr->_flags._tran = (trans < 0) ? !spr->_flags._tran : (trans != 0); } void CGEEngine::snPort(Sprite *spr, int port) { debugC(1, kCGEDebugEngine, "CGEEngine::snPort(spr, %d)", port); if (spr) spr->_flags._port = (port < 0) ? !spr->_flags._port : (port != 0); } void CGEEngine::snKill(Sprite *spr) { debugC(1, kCGEDebugEngine, "CGEEngine::snKill(spr)"); if (!spr) return; if (spr->_flags._kept) { int n = findPocket(spr); if (n >= 0) _pocket[n] = nullptr; } Sprite *nx = spr->_next; hide1(spr); _vga->_showQ->remove(spr); _eventManager->clearEvent(spr); if (spr->_flags._kill) { delete spr; } else { spr->_scene = -1; _vga->_spareQ->append(spr); } if (nx) { if (nx->_flags._slav) snKill(nx); } } /** * Play a FX sound * @param spr Sprite pointer * @param wav FX index */ void CGEEngine::snSound(Sprite *spr, int wav) { debugC(1, kCGEDebugEngine, "CGEEngine::snSound(spr, %d)", wav); if (wav == -1) _sound->stop(); else _sound->play((*_fx)[wav], (spr) ? ((spr->_x + spr->_w / 2) / (kScrWidth / 16)) : 8); _sound->setRepeat(1); } void CGEEngine::snKeep(Sprite *spr, int stp) { debugC(1, kCGEDebugEngine, "CGEEngine::snKeep(spr, %d)", stp); selectPocket(-1); if (spr && ! spr->_flags._kept && _pocket[_pocPtr] == nullptr) { int16 oldRepeat = _sound->getRepeat(); _sound->setRepeat(1); snSound(spr, 3); _sound->setRepeat(oldRepeat); _pocket[_pocPtr] = spr; spr->_scene = 0; spr->_flags._kept = true; spr->gotoxy(kPocketX + kPocketDX * _pocPtr + kPocketDX / 2 - spr->_w / 2, kPocketY + kPocketDY / 2 - spr->_h / 2); if (stp >= 0) spr->step(stp); } selectPocket(-1); } /** * Remove an object from the inventory and (if specified) trigger an animation * @param spr Inventory item * @param stp Animation */ void CGEEngine::snGive(Sprite *spr, int stp) { debugC(1, kCGEDebugEngine, "CGEEngine::snGive(spr, %d)", stp); if (spr) { int p = findPocket(spr); if (p >= 0) { _pocket[p] = nullptr; spr->_scene = _now; spr->_flags._kept = false; if (stp >= 0) spr->step(stp); } } selectPocket(-1); } void CGEEngine::snBackPt(Sprite *spr, int stp) { debugC(1, kCGEDebugEngine, "CGEEngine::snBackPt(spr, %d)", stp); if (spr) { if (stp >= 0) spr->step(stp); spr->backShow(true); } } void CGEEngine::snLevel(Sprite *spr, int lev) { debugC(1, kCGEDebugEngine, "CGEEngine::snLevel(spr, %d)", lev); assert((lev >= 0) && (lev < 5)); for (int i = 0; i < 5; i++) { spr = _vga->_spareQ->locate(100 + i); if (spr) { if (i <= lev) { spr->backShow(true); spr->_scene = 0; spr->_flags._hide = false; } else { spr->_flags._hide = true; spr->_scene = -1; } } else { warning("SPR not found! ref: %d", 100 + i); } } _lev = lev; _maxScene = _maxSceneArr[_lev]; } /** * Set a flag to a value * @param indx Flag index * @param val Flag value */ void CGEEngine::snFlag(int indx, bool val) { _flag[indx] = val; } void CGEEngine::snSetRef(Sprite *spr, int nr) { debugC(1, kCGEDebugEngine, "CGEEngine::snSetRef(spr, %d)", nr); if (spr) spr->_ref = nr; } void CGEEngine::snFlash(bool on) { debugC(1, kCGEDebugEngine, "CGEEngine::snFlash(%s)", on ? "true" : "false"); if (on) { Dac *pal = (Dac *)malloc(sizeof(Dac) * kPalCount); if (pal) { memcpy(pal, _vga->_sysPal, kPalSize); for (int i = 0; i < kPalCount; i++) { int c; c = pal[i]._r << 1; pal[i]._r = (c < 64) ? c : 63; c = pal[i]._g << 1; pal[i]._g = (c < 64) ? c : 63; c = pal[i]._b << 1; pal[i]._b = (c < 64) ? c : 63; } _vga->setColors(pal, 64); } free(pal); } else _vga->setColors(_vga->_sysPal, 64); _dark = false; } void CGEEngine::snLight(bool in) { debugC(1, kCGEDebugEngine, "CGEEngine::snLight(%s)", in ? "true" : "false"); if (in) _vga->sunrise(_vga->_sysPal); else _vga->sunset(); _dark = !in; } /** * Set an horizontal boundary * @param scene Scene number * @param barX Horizontal boundary value */ void CGEEngine::snHBarrier(const int scene, const int barX) { debugC(1, kCGEDebugEngine, "CGEEngine::snHBarrier(%d, %d)", scene, barX); _barriers[(scene > 0) ? scene : _now]._horz = barX; } /** * Set a vertical boundary * @param scene Scene number * @param barY Vertical boundary value */ void CGEEngine::snVBarrier(const int scene, const int barY) { debugC(1, kCGEDebugEngine, "CGEEngine::snVBarrier(%d, %d)", scene, barY); _barriers[(scene > 0) ? scene : _now]._vert = barY; } void CGEEngine::snWalk(Sprite *spr, int x, int y) { debugC(1, kCGEDebugEngine, "CGEEngine::snWalk(spr, %d, %d)", x, y); if (_hero) { if (spr && y < 0) _hero->findWay(spr); else _hero->findWay(XZ(x, y)); } } void CGEEngine::snReach(Sprite *spr, int mode) { debugC(1, kCGEDebugEngine, "CGEEngine::snReach(spr, %d)", mode); if (_hero) _hero->reach(spr, mode); } void CGEEngine::snMouse(bool on) { debugC(1, kCGEDebugEngine, "CGEEngine::snMouse(%s)", on ? "true" : "false"); if (on) _mouse->on(); else _mouse->off(); } } // End of namespace CGE