scummvm/engines/kyra/sequences_hof.cpp
Florian Kagerer 0efebd9876 - add support for HOF floppy versions
- playing directly from installer files (WESTWOOD.00X) is supported (not recommended for low-end machines and certain ports)

svn-id: r32249
2008-05-24 16:30:18 +00:00

2816 lines
73 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.
*
* $URL$
* $Id$
*
*/
#include "kyra/kyra_v1.h"
#include "kyra/kyra_hof.h"
#include "kyra/screen.h"
#include "kyra/wsamovie.h"
#include "kyra/sound.h"
#include "kyra/text_hof.h"
#include "kyra/timer.h"
#include "kyra/resource.h"
#include "common/system.h"
namespace Kyra {
void KyraEngine_HoF::seq_playSequences(int startSeq, int endSeq) {
debugC(9, kDebugLevelMain, "KyraEngine_HoF::seq_playSequences(%i, %i)", startSeq, endSeq);
seq_init();
bool allowSkip = (!(_flags.isDemo && !_flags.isTalkie) && (startSeq == kSequenceTitle)) ? false : true;
if (endSeq == -1)
endSeq = startSeq;
assert(startSeq >= 0 && endSeq < kSequenceArraySize && startSeq <= endSeq);
_sound->setSoundList(&_soundData[(startSeq > kSequenceZanfaun) ? kMusicFinale : kMusicIntro]);
_sound->loadSoundFile(0);
_screen->_charWidth = -2;
memset(_activeWSA, 0, sizeof(ActiveWSA) * 8);
for (int i = 0; i < 8; ++i)
_activeWSA[i].flags = -1;
memset(_activeText, 0, sizeof(ActiveText) * 10);
seq_resetAllTextEntries();
_screen->hideMouse();
int oldPage = _screen->setCurPage(2);
for (int i = 0; i < 4; ++i)
memset(_screen->getPalette(i), 0, 0x300);
_screen->clearPage(10);
_screen->clearPage(12);
_seqSubframePlaying = false;
_seqWsaCurrentFrame = 0;
_seqTextColor[0] = _seqTextColor[1] = 0;
_seqEndTime = 0;
_menuChoice = 0;
for (int seqNum = startSeq; seqNum <= endSeq && !((skipFlag() && allowSkip) || _quitFlag || (_abortIntroFlag && allowSkip) || _menuChoice); seqNum++) {
_screen->clearPage(0);
_screen->clearPage(8);
memcpy(_screen->getPalette(1), _screen->getPalette(0), 0x300);
_seqFrameCounter = 0;
_seqStartTime = _system->getMillis();
allowSkip = (!(_flags.isDemo && !_flags.isTalkie) && (seqNum == kSequenceTitle)) ? false : true;
Sequence cseq = _sequences->seq[seqNum];
SeqProc cb = _callbackS[seqNum];
if (cseq.flags & 2) {
_screen->loadBitmap(cseq.cpsFile, 2, 2, _screen->getPalette(0));
_screen->setScreenPalette(_screen->getPalette(0));
} else {
_screen->setCurPage(2);
_screen->clearPage(2);
_screen->loadPalette("goldfont.col", _screen->getPalette(0));
}
if (cb && !(_flags.isDemo && !_flags.isTalkie))
(this->*cb)(0, 0, 0, -1);
if (cseq.flags & 1) {
_seqWsa->close();
_seqWsa->open(cseq.wsaFile, 0, _screen->getPalette(0));
_screen->setScreenPalette(_screen->getPalette(0));
_seqWsa->setX(cseq.xPos);
_seqWsa->setY(cseq.yPos);
_seqWsa->setDrawPage(2);
_seqWsa->displayFrame(0, 0);
}
if (cseq.flags & 4) {
int cp = _screen->setCurPage(2);
Screen::FontId cf = _screen->setFont(Screen::FID_GOLDFONT_FNT);
if (cseq.stringIndex1 != -1) {
int sX = (320 - _screen->getTextWidth(_sequenceStrings[cseq.stringIndex1])) / 2;
_screen->printText(_sequenceStrings[cseq.stringIndex1], sX, 100 - _screen->getFontHeight(), 1, 0);
}
if (cseq.stringIndex2 != -1) {
int sX = (320 - _screen->getTextWidth(_sequenceStrings[cseq.stringIndex2])) / 2;
_screen->printText(_sequenceStrings[cseq.stringIndex2], sX, 100, 1, 0);
}
_screen->setFont(cf);
_screen->setCurPage(cp);
}
_screen->copyPage(2, 12);
_screen->copyPage(0, 2);
_screen->copyPage(2, 10);
_screen->copyPage(12, 2);
seq_sequenceCommand(cseq.startupCommand);
if (!((skipFlag() && allowSkip) || _quitFlag || (_abortIntroFlag && allowSkip) || _menuChoice)) {
_screen->copyPage(2, 0);
_screen->updateScreen();
}
if (cseq.flags & 1) {
int w2 = _seqWsa->width();
int h2 = _seqWsa->height();
int x = cseq.xPos;
int y = cseq.yPos;
_seqFrameDelay = cseq.frameDelay;
if (_seqWsa) {
if (x < 0) {
x = 0;
w2 = 0;
}
if (y < 0) {
y = 0;
h2 = 0;
}
if (cseq.xPos + _seqWsa->width() > 319)
_seqWsa->setWidth(320 - cseq.xPos);
if (cseq.yPos + _seqWsa->height() > 199)
_seqWsa->setHeight(199 - cseq.yPos);
}
uint8 dir = (cseq.startFrame > cseq.numFrames) ? 0 : 1;
_seqWsaCurrentFrame = cseq.startFrame;
bool loop = true;
while (loop && !((skipFlag() && allowSkip) || _quitFlag || (_abortIntroFlag && allowSkip) || _menuChoice)) {
_seqEndTime = _system->getMillis() + _seqFrameDelay * _tickLength;
if (_seqWsa || !cb)
_screen->copyPage(12, 2);
if (cb) {
int f = _seqWsaCurrentFrame % _seqWsa->frames();
(this->*cb)(_seqWsa, cseq.xPos, cseq.yPos, f);
}
if (_seqWsa) {
int f = _seqWsaCurrentFrame % _seqWsa->frames();
_seqWsa->setX(cseq.xPos);
_seqWsa->setY(cseq.yPos);
_seqWsa->setDrawPage(2);
_seqWsa->displayFrame(f, 0);
}
_screen->copyPage(2, 12);
seq_processWSAs();
seq_processText();
if ((_seqWsa || !cb) && !((skipFlag() && allowSkip) || _quitFlag || (_abortIntroFlag && allowSkip) || _menuChoice)) {
_screen->copyPage(2, 0);
_screen->updateScreen();
}
bool loop2 = true;
while (loop2 && !((skipFlag() && allowSkip) || _quitFlag || (_abortIntroFlag && allowSkip) || _menuChoice)) {
if (_seqWsa) {
seq_processText();
if (!((skipFlag() && allowSkip) || _quitFlag || (_abortIntroFlag && allowSkip) || _menuChoice)) {
_screen->copyPage(2, 0);
_screen->updateScreen();
}
uint32 now = _system->getMillis();
if (now >= _seqEndTime) {
loop2 = false;
} else {
uint32 tdiff = _seqEndTime - now;
uint32 dly = tdiff < _tickLength ? tdiff : _tickLength;
delay(dly);
}
} else {
loop = loop2 = false;
}
}
if (loop) {
if (dir == 1) {
if (++_seqWsaCurrentFrame >= cseq.numFrames)
loop = false;
} else {
if (--_seqWsaCurrentFrame < cseq.numFrames)
loop = false;
}
}
}
_seqWsa->close();
} else {
_seqFrameDelay = cseq.frameDelay;
_seqEndTime = _system->getMillis() + _seqFrameDelay * _tickLength;
while (!((skipFlag() && allowSkip) || _quitFlag || (_abortIntroFlag && allowSkip) || _menuChoice)) {
uint32 starttime = _system->getMillis();
seq_processWSAs();
if (cb)
(this->*cb)(0, 0, 0, 0);
seq_processText();
_screen->copyPage(2, 0);
_screen->updateScreen();
_screen->copyPage(12, 2);
uint32 now = _system->getMillis();
if (now >= _seqEndTime && !_seqSubframePlaying)
break;
uint32 tdiff = _seqEndTime - starttime;
int32 dly = _tickLength - (now - starttime);
if (dly > 0)
delay(MIN<uint32>(dly, tdiff));
}
}
if (cb && !(_flags.isDemo && !_flags.isTalkie))
(this->*cb)(0, 0, 0, -2);
uint32 ct = seq_activeTextsTimeLeft();
uint32 dl = cseq.duration * _tickLength;
if (dl < ct)
dl = ct;
_seqEndTime = _system->getMillis() + dl;
while (!((skipFlag() && allowSkip) || _quitFlag || (_abortIntroFlag && allowSkip) || _menuChoice)) {
uint32 starttime = _system->getMillis();
seq_processWSAs();
_screen->copyPage(2, 0);
_screen->updateScreen();
_screen->copyPage(12, 2);
uint32 now = _system->getMillis();
if (now >= _seqEndTime && !_seqSubframePlaying) {
break;
}
uint32 tdiff = _seqEndTime - starttime;
int32 dly = _tickLength - (now - starttime);
if (dly > 0)
delay(MIN<uint32>(dly, tdiff));
}
seq_sequenceCommand(cseq.finalCommand);
seq_resetAllTextEntries();
if (_abortIntroFlag || skipFlag()) {
_sound->haltTrack();
_sound->voiceStop();
}
if (_flags.isDemo && !_flags.isTalkie) {
if (seqNum == kSequenceDemoFisher) {
_abortIntroFlag = false;
resetSkipFlag();
seqNum = kSequenceDemoVirgin;
}
} else {
if ((seqNum != kSequenceTitle && seqNum < kSequenceZanfaun &&
(_abortIntroFlag || skipFlag())) || seqNum == kSequenceZanfaun) {
_abortIntroFlag = false;
_eventList.clear();
seqNum = kSequenceWestwood;
} else if (seqNum < kSequenceFrash && (_abortIntroFlag || skipFlag())) {
_abortIntroFlag = false;
_eventList.clear();
seqNum = kSequenceFirates;
}
}
if (_menuChoice) {
_abortIntroFlag = false;
_eventList.clear();
if (_menuChoice == 2) {
seqNum = kSequenceTitle;
_menuChoice = 0;
}
}
}
if (!_menuChoice)
delay(1000);
_screen->setCurPage(oldPage);
_screen->showMouse();
for (int i = 0; i < 8; i++)
seq_unloadWSA(i);
_seqWsa->close();
_screen->_charWidth = 0;
seq_uninit();
}
int KyraEngine_HoF::seq_introWestwood(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
debugC(9, kDebugLevelMain, "KyraEngine_HoF::seq_introWestwood(%p, %i, %i, %i)", (const void*)wsaObj, x, y, frm);
if (frm == -2) {
if (_flags.platform == Common::kPlatformFMTowns || _flags.platform == Common::kPlatformPC98)
delay(300 * _tickLength);
} else if (!frm) {
_sound->playTrack(2);
}
return 0;
}
int KyraEngine_HoF::seq_introTitle(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
debugC(9, kDebugLevelMain, "KyraEngine_HoF::seq_introTitle(%p, %i, %i, %i)", (const void*)wsaObj, x, y, frm);
if (frm == 1) {
_sound->playTrack(3);
} else if (frm == 25) {
int cp = _screen->setCurPage(0);
_screen->showMouse();
_system->updateScreen();
_menuChoice = _menu->handle(11) + 1;
_seqEndTime = 0;
_seqSubframePlaying = false;
if (_menuChoice == 4)
quitGame();
_screen->hideMouse();
_screen->setCurPage(cp);
}
return 0;
}
int KyraEngine_HoF::seq_introOverview(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
debugC(9, kDebugLevelMain, "KyraEngine_HoF::seq_introOverview(%p, %i, %i, %i)", (const void*)wsaObj, x, y, frm);
uint8 *tmpPal = &(_screen->getPalette(3)[0x101]);
memset(tmpPal, 0, 256);
uint32 endtime = 0, now = 0;
switch (_seqFrameCounter) {
case 0:
_seqSubframePlaying = true;
_sound->playTrack(4);
endtime = _system->getMillis() + 60 * _tickLength;
_seqTextColor[1] = _screen->findLeastDifferentColor(_seqTextColorPresets, _screen->getPalette(0) + 3, 255) & 0xff;
memset(_seqTextColorMap, _seqTextColor[1], 16);
_seqTextColorMap[1] = _seqTextColor[0] = _screen->findLeastDifferentColor(_seqTextColorPresets + 3, _screen->getPalette(0) + 3, 255) & 0xff;
_screen->setTextColorMap(_seqTextColorMap);
now = _system->getMillis();
if (endtime > now)
delay(endtime - now);
break;
case 1:
_screen->generateGrayOverlay(_screen->getPalette(0), _screen->getPalette(3), 0x40, 0, 0, 0, 0x100, true);
for (int i = 0; i < 256; i++)
tmpPal[_screen->getPalette(3)[i]] = 1;
for (int i = 0; i < 256; i++) {
int v = (tmpPal[i] == 1) ? i : _screen->getPalette(3)[i];
v *= 3;
_screen->getPalette(2)[3 * i] = _screen->getPalette(0)[v];
_screen->getPalette(2)[3 * i + 1] = _screen->getPalette(0)[v + 1];
_screen->getPalette(2)[3 * i + 2] = _screen->getPalette(0)[v + 2];
}
break;
case 40:
seq_loadNestedSequence(0, kSequenceOver1);
break;
case 60:
seq_loadNestedSequence(1, kSequenceOver2);
break;
case 120:
seq_playTalkText(0);
break;
case 200:
seq_waitForTextsTimeout();
_screen->fadePalette(_screen->getPalette(2), 64);
break;
case 201:
_screen->setScreenPalette(_screen->getPalette(2));
_screen->updateScreen();
_screen->applyOverlay(0, 0, 320, 200, 2, _screen->getPalette(3));
_screen->copyPage(2, 12);
_screen->copyRegion(0, 0, 0, 0, 320, 200, 2, 0);
_screen->setScreenPalette(_screen->getPalette(0));
_screen->updateScreen();
seq_resetActiveWSA(0);
seq_resetActiveWSA(1);
break;
case 282:
seq_loadNestedSequence(0, kSequenceForest);
seq_playTalkText(1);
break;
case 354:
case 434:
if (!((_seqFrameCounter == 354 && (_flags.platform == Common::kPlatformFMTowns || _flags.platform == Common::kPlatformPC98)) || (_seqFrameCounter == 434 && _flags.platform == Common::kPlatformPC)))
break;
seq_resetActiveWSA(0);
seq_loadNestedSequence(0, kSequenceDragon);
break;
case 400:
case 540:
if (!((_seqFrameCounter == 400 && (_flags.platform == Common::kPlatformFMTowns || _flags.platform == Common::kPlatformPC98)) || (_seqFrameCounter == 540 && _flags.platform == Common::kPlatformPC)))
break;
seq_waitForTextsTimeout();
seq_resetActiveWSA(0);
_seqEndTime = 0;
_seqSubframePlaying = false;
break;
default:
break;
}
_seqFrameCounter++;
return 0;
}
int KyraEngine_HoF::seq_introLibrary(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
debugC(9, kDebugLevelMain, "KyraEngine_HoF::seq_introLibrary(%p, %i, %i, %i)", (const void*)wsaObj, x, y, frm);
switch (_seqFrameCounter) {
case 0:
_seqSubframePlaying = true;
_sound->playTrack(5);
_screen->generateGrayOverlay(_screen->getPalette(0), _screen->getPalette(3), 0x24, 0, 0, 0, 0x100, false);
_seqTextColor[1] = _screen->findLeastDifferentColor(_seqTextColorPresets, _screen->getPalette(0) + 3, 255) & 0xff;
memset(_seqTextColorMap, _seqTextColor[1], 16);
_seqTextColorMap[1] = _seqTextColor[0] = _screen->findLeastDifferentColor(_seqTextColorPresets + 3, _screen->getPalette(0) + 3, 255) & 0xff;
_screen->setTextColorMap(_seqTextColorMap);
break;
case 1:
seq_loadNestedSequence(0, kSequenceLibrary3);
seq_playTalkText(4);
break;
case 100:
seq_waitForTextsTimeout();
_screen->copyPage(12, 2);
_screen->applyOverlay(0, 0, 320, 200, 2, _screen->getPalette(3));
_screen->copyRegion(0, 0, 0, 0, 320, 200, 2, 0);
_screen->updateScreen();
_screen->copyPage(2, 12);
seq_resetActiveWSA(0);
seq_loadNestedSequence(0, kSequenceDarm);
break;
case 104:
seq_playTalkText(5);
break;
case 240:
seq_waitForTextsTimeout();
seq_resetActiveWSA(0);
seq_loadNestedSequence(0, kSequenceLibrary2);
break;
case 340:
seq_resetActiveWSA(0);
_screen->applyOverlay(0, 0, 320, 200, 2, _screen->getPalette(3));
_screen->copyPage(2, 12);
_screen->copyRegion(0, 0, 0, 0, 320, 200, 2, 0);
_screen->updateScreen();
seq_loadNestedSequence(0, kSequenceMarco);
seq_playTalkText(6);
break;
case 480:
case 660:
if (!((_seqFrameCounter == 480 && (_flags.platform == Common::kPlatformFMTowns || _flags.platform == Common::kPlatformPC98)) || (_seqFrameCounter == 660 && _flags.platform == Common::kPlatformPC)))
break;
_screen->copyPage(2, 12);
seq_waitForTextsTimeout();
seq_resetActiveWSA(0);
_seqEndTime = 0;
_seqSubframePlaying = false;
break;
default:
break;
}
_seqFrameCounter++;
return 0;
}
int KyraEngine_HoF::seq_introHand(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
debugC(9, kDebugLevelMain, "KyraEngine_HoF::seq_introHand(%p, %i, %i, %i)", (const void*)wsaObj, x, y, frm);
switch (_seqFrameCounter) {
case 0:
_seqSubframePlaying = true;
_sound->playTrack(6);
_screen->generateGrayOverlay(_screen->getPalette(0), _screen->getPalette(3), 0x24, 0, 0, 0, 0x100, false);
_seqTextColor[1] = _screen->findLeastDifferentColor(_seqTextColorPresets, _screen->getPalette(0) + 3, 255) & 0xff;
memset(_seqTextColorMap, _seqTextColor[1], 16);
_seqTextColorMap[1] = _seqTextColor[0] = _screen->findLeastDifferentColor(_seqTextColorPresets + 3, _screen->getPalette(0) + 3, 255) & 0xff;
_screen->setTextColorMap(_seqTextColorMap);
break;
case 1:
seq_loadNestedSequence(0, kSequenceHand1a);
seq_loadNestedSequence(1, kSequenceHand1b);
seq_loadNestedSequence(2, kSequenceHand1c);
seq_playTalkText(7);
break;
case 201:
seq_waitForTextsTimeout();
_screen->applyOverlay(0, 0, 320, 200, 2, _screen->getPalette(3));
_screen->copyPage(2, 12);
_screen->copyRegion(0, 0, 0, 0, 320, 200, 2, 0);
_screen->updateScreen();
seq_resetActiveWSA(0);
seq_resetActiveWSA(1);
seq_resetActiveWSA(2);
seq_loadNestedSequence(0, kSequenceHand2);
seq_playTalkText(8);
break;
case 260:
case 395:
if (!((_seqFrameCounter == 260 && (_flags.platform == Common::kPlatformFMTowns || _flags.platform == Common::kPlatformPC98)) || (_seqFrameCounter == 395 && _flags.platform == Common::kPlatformPC)))
break;
seq_waitForTextsTimeout();
seq_resetActiveWSA(0);
seq_loadNestedSequence(1, kSequenceHand3);
seq_playTalkText(9);
break;
case 365:
case 500:
if (!((_seqFrameCounter == 365 && (_flags.platform == Common::kPlatformFMTowns || _flags.platform == Common::kPlatformPC98)) || (_seqFrameCounter == 500 && _flags.platform == Common::kPlatformPC)))
break;
seq_waitForTextsTimeout();
seq_resetActiveWSA(1);
seq_loadNestedSequence(0, kSequenceHand4);
break;
case 405:
case 540:
if (!((_seqFrameCounter == 405 && (_flags.platform == Common::kPlatformFMTowns || _flags.platform == Common::kPlatformPC98)) || (_seqFrameCounter == 540 && _flags.platform == Common::kPlatformPC)))
break;
seq_playTalkText(10);
break;
case 484:
case 630:
if (!((_seqFrameCounter == 484 && (_flags.platform == Common::kPlatformFMTowns || _flags.platform == Common::kPlatformPC98)) || (_seqFrameCounter == 630 && _flags.platform == Common::kPlatformPC)))
break;
seq_waitForTextsTimeout();
seq_resetActiveWSA(0);
_seqEndTime = 0;
_seqSubframePlaying = false;
break;
default:
break;
}
_seqFrameCounter++;
return 0;
}
int KyraEngine_HoF::seq_introPoint(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (frm == -2) {
seq_waitForTextsTimeout();
_seqEndTime = 0;
}
switch (_seqFrameCounter) {
case -2:
seq_waitForTextsTimeout();
break;
case 0:
_sound->playTrack(7);
_seqTextColor[1] = 0xf7;
memset(_seqTextColorMap, _seqTextColor[1], 16);
_seqTextColorMap[1] = _seqTextColor[0] = _screen->findLeastDifferentColor(_seqTextColorPresets + 3, _screen->getPalette(0) + 3, 255) & 0xff;
_screen->setTextColorMap(_seqTextColorMap);
_screen->generateGrayOverlay(_screen->getPalette(0), _screen->getPalette(3), 0x24, 0, 0, 0, 0x100, false);
break;
case 1:
seq_playTalkText(11);
break;
default:
break;
}
_seqFrameCounter++;
return 0;
}
int KyraEngine_HoF::seq_introZanfaun(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (frm == -2) {
seq_waitForTextsTimeout();
_seqEndTime = 0;
return 0;
}
switch (_seqFrameCounter) {
case 0:
_sound->playTrack(8);
_seqTextColor[1] = 0xfd;
memset(_seqTextColorMap, _seqTextColor[1], 16);
_seqTextColorMap[1] = _seqTextColor[0] = _screen->findLeastDifferentColor(_seqTextColorPresets + 3, _screen->getPalette(0) + 3, 255) & 0xff;
_screen->setTextColorMap(_seqTextColorMap);
break;
case 1:
if (_flags.isTalkie) {
seq_playWsaSyncDialogue(21, 13, -1, 140, 70, 160, wsaObj, 0, 8, x, y);
} else {
seq_setTextEntry(21, 140, 70, 200, 160);
_seqFrameDelay = 200;
}
break;
case 2:
case 11:
case 21:
if (!_flags.isTalkie)
_seqFrameDelay = 12;
break;
case 9:
if (_flags.isTalkie)
seq_playWsaSyncDialogue(13, 14, -1, 140, (_flags.lang == Common::FR_FRA
|| _flags.lang == Common::DE_DEU) ? 50 : 70, 160, wsaObj, 9, 15, x, y);
break;
case 10:
if (!_flags.isTalkie) {
seq_waitForTextsTimeout();
seq_setTextEntry(13, 140, 50, _sequenceStringsDuration[13], 160);
_seqFrameDelay = 300;
}
break;
case 16:
if (_flags.isTalkie)
seq_playWsaSyncDialogue(18, 15, -1, 140, (_flags.lang == Common::FR_FRA) ? 50 :
(_flags.lang == Common::DE_DEU ? 40 : 70), 160, wsaObj, 10, 16, x, y);
break;
case 17:
if (_flags.isTalkie)
_seqFrameDelay = 12;
break;
case 20:
if (!_flags.isTalkie) {
seq_waitForTextsTimeout();
seq_setTextEntry(18, 160, 50, _sequenceStringsDuration[18], 160);
_seqFrameDelay = 200;
}
break;
case 26:
seq_waitForTextsTimeout();
break;
case 46:
if (_flags.isTalkie) {
seq_playWsaSyncDialogue(16, 16, -1, 200, 50, 120, wsaObj, 46, 46, x, y);
} else {
seq_waitForTextsTimeout();
seq_setTextEntry(16, 200, 50, _sequenceStringsDuration[16], 120);
}
_seqEndTime = _system->getMillis() + 120 * _tickLength;
break;
default:
break;
}
_seqFrameCounter++;
return 0;
}
int KyraEngine_HoF::seq_introOver1(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (frm == 2)
seq_waitForTextsTimeout();
else if (frm == 3)
seq_playTalkText(12);
return frm;
}
int KyraEngine_HoF::seq_introOver2(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (frm == 1)
seq_playTalkText(12);
return frm;
}
int KyraEngine_HoF::seq_introForest(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (frm == 11)
seq_waitForTextsTimeout();
else if (frm == 12)
seq_playTalkText(2);
return frm;
}
int KyraEngine_HoF::seq_introDragon(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (frm == 11)
seq_waitForTextsTimeout();
else if (frm == 3)
seq_playTalkText(3);
return frm;
}
int KyraEngine_HoF::seq_introDarm(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
//NULLSUB (at least in fm-towns version)
return frm;
}
int KyraEngine_HoF::seq_introLibrary2(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
//NULLSUB (at least in fm-towns version)
return frm;
}
int KyraEngine_HoF::seq_introMarco(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (frm == 36) {
seq_waitForTextsTimeout();
_seqEndTime = 0;
}
return frm;
}
int KyraEngine_HoF::seq_introHand1a(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
//NULLSUB (at least in fm-towns version)
return frm;
}
int KyraEngine_HoF::seq_introHand1b(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (frm == 15)
frm = 12;
return frm;
}
int KyraEngine_HoF::seq_introHand1c(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (frm == 8)
frm = 4;
return frm;
}
int KyraEngine_HoF::seq_introHand2(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
//NULLSUB (at least in fm-towns version)
return frm;
}
int KyraEngine_HoF::seq_introHand3(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
//NULLSUB (at least in fm-towns version)
return frm;
}
int KyraEngine_HoF::seq_finaleFunters(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
uint32 endtime = 0;
int chatX = 0;
int chatY = 0;
int chatW = 0;
int chatFirstFrame = 0;
int chatLastFrame = 0;
uint16 voiceIndex = 0;
switch (frm) {
case -2:
seq_sequenceCommand(9);
break;
case 0:
_sound->playTrack(3);
_seqTextColor[1] = _screen->findLeastDifferentColor(_seqTextColorPresets, _screen->getPalette(0) + 3, 255) & 0xff;
memset(_seqTextColorMap, _seqTextColor[1], 16);
_seqTextColor[0] = _seqTextColorMap[1] = 0xff;
_screen->setTextColorMap(_seqTextColorMap);
endtime = _system->getMillis() + 480 * _tickLength;
seq_printCreditsString(81, 240, 70, _seqTextColorMap, 252);
seq_printCreditsString(82, 240, 90, _seqTextColorMap, _seqTextColor[0]);
_screen->copyPage(2, 12);
seq_playTalkText(_flags.isTalkie ? 28 : 24);
delay(endtime - _system->getMillis());
_seqTextColor[0] = 1;
if (_flags.isTalkie) {
chatY = (_flags.lang == Common::FR_FRA) ? 70 : 78;
chatFirstFrame = 9;
chatLastFrame = 15;
voiceIndex = 34;
} else {
chatY = (_flags.lang == Common::FR_FRA) ? 78 : 70;
chatFirstFrame = 0;
chatLastFrame = 8;
}
chatX = (_flags.lang == Common::FR_FRA) ? 84 : 88;
chatW = 100;
seq_playWsaSyncDialogue(22, voiceIndex, 187, chatX, chatY, chatW, wsaObj, chatFirstFrame, chatLastFrame, x, y);
break;
case 9:
case 16:
if (!((frm == 9 && !_flags.isTalkie) || (frm == 16 && _flags.isTalkie)))
break;
_seqFrameDelay = 12;
if (_flags.lang == Common::FR_FRA) {
chatX = 80;
chatW = 112;
} else {
chatX = (_flags.lang == Common::DE_DEU) ? 84 : 96;
chatW = 100;
}
if (_flags.isTalkie) {
chatFirstFrame = 0;
chatLastFrame = 8;
voiceIndex = 35;
} else {
chatFirstFrame = 9;
chatLastFrame = 15;
}
chatY = 70;
seq_playWsaSyncDialogue(23, voiceIndex, 137, chatX, chatY, chatW, wsaObj, chatFirstFrame, chatLastFrame, x, y);
if (_flags.isTalkie)
_seqWsaCurrentFrame = 17;
break;
default:
break;
}
_seqFrameCounter++;
return 0;
}
int KyraEngine_HoF::seq_finaleFerb(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
uint32 endtime = 0;
int chatX = 0;
int chatY = 0;
int chatW = 0;
int chatFirstFrame = 0;
int chatLastFrame = 0;
uint16 voiceIndex = 0;
switch (frm) {
case -2:
seq_sequenceCommand(9);
endtime = _system->getMillis() + 480 * _tickLength;
seq_printCreditsString(34, 240, _flags.isTalkie ? 60 : 40, _seqTextColorMap, 252);
seq_printCreditsString(35, 240, _flags.isTalkie ? 70 : 50, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(36, 240, _flags.isTalkie ? 90 : 70, _seqTextColorMap, 252);
seq_printCreditsString(37, 240, _flags.isTalkie ? 100 : 90, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(38, 240, _flags.isTalkie ? 120 : 110, _seqTextColorMap, 252);
seq_printCreditsString(39, 240, _flags.isTalkie ? 130 : 120, _seqTextColorMap, _seqTextColor[0]);
if (_flags.platform == Common::kPlatformFMTowns || _flags.platform == Common::kPlatformPC98)
seq_printCreditsString(103, 240, 130, _seqTextColorMap, _seqTextColor[0]);
delay(endtime - _system->getMillis());
_seqEndTime = 0;
break;
case 0:
_seqTextColor[1] = _screen->findLeastDifferentColor(_seqTextColorPresets, _screen->getPalette(0) + 3, 255) & 0xff;
memset(_seqTextColorMap, _seqTextColor[1], 16);
_seqTextColor[0] = _seqTextColorMap[1] = 255;
_screen->setTextColorMap(_seqTextColorMap);
break;
case 5:
if (!_flags.isTalkie)
seq_playTalkText(18);
_seqFrameDelay = 16;
if (_flags.isTalkie) {
chatFirstFrame = 5;
chatLastFrame = 8;
voiceIndex = 22;
} else {
chatLastFrame = 14;
}
chatX = 116;
chatY = 90;
chatW = 60;
seq_playWsaSyncDialogue(24, voiceIndex, 149, chatX, chatY, chatW, wsaObj, chatFirstFrame, chatLastFrame, x, y);
break;
case 11:
if (_flags.isTalkie)
seq_playWsaSyncDialogue(24, 22, 149, 116, 90, 60, wsaObj, 11, 14, x, y);
break;
case 16:
seq_playTalkText(_flags.isTalkie ? 23 : 19);
_seqFrameDelay = _flags.isTalkie ? 20 : 16;
if (_flags.lang == Common::FR_FRA) {
chatY = 48;
chatW = 88;
} else {
chatY = 60;
chatW = 100;
}
chatX = 60;
if (_flags.isTalkie)
voiceIndex = 36;
seq_playWsaSyncDialogue(25, voiceIndex, 143, chatX, chatY, chatW, wsaObj, 16, 25, x, y);
_seqFrameDelay = 16;
break;
default:
break;
}
_seqFrameCounter++;
return 0;
}
int KyraEngine_HoF::seq_finaleFish(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
uint32 endtime = 0;
int chatX = 0;
int chatY = 0;
int chatW = 0;
uint16 voiceIndex = 0;
switch (frm) {
case -2:
seq_sequenceCommand(9);
endtime = _system->getMillis() + 480 * _tickLength;
seq_printCreditsString(40, 240, _flags.isTalkie ? 55 : 40, _seqTextColorMap, 252);
seq_printCreditsString(41, 240, _flags.isTalkie ? 65 : 50, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(42, 240, _flags.isTalkie ? 75 : 60, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(43, 240, _flags.isTalkie ? 95 : 80, _seqTextColorMap, 252);
seq_printCreditsString(44, 240, _flags.isTalkie ? 105 : 90, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(93, 240, _flags.isTalkie ? 125 : 110, _seqTextColorMap, 252);
seq_printCreditsString(94, 240, _flags.isTalkie ? 135 : 120, _seqTextColorMap, _seqTextColor[0]);
delay(endtime - _system->getMillis());
_seqEndTime = 0;
break;
case 0:
_seqTextColor[1] = _screen->findLeastDifferentColor(_seqTextColorPresets, _screen->getPalette(0) + 3, 255) & 0xff;
memset(_seqTextColorMap, _seqTextColor[1], 16);
_seqTextColor[0] = _seqTextColorMap[1] = 0xff;
_screen->setTextColorMap(_seqTextColorMap);
break;
case 4:
chatX = 94;
chatY = 42;
chatW = 100;
if (_flags.isTalkie)
voiceIndex = 37;
seq_playWsaSyncDialogue(26, voiceIndex, 149, chatX, chatY, chatW, wsaObj, 3, 12, x, y);
break;
case 14:
seq_playTalkText(_flags.isTalkie ? 19 : 15);
break;
case 23:
seq_playTalkText(_flags.isTalkie ? 20 : 16);
break;
case 29:
chatX = (_flags.lang == Common::DE_DEU) ? 82 : ((_flags.lang == Common::FR_FRA) ? 92 : 88);
chatY = 40;
chatW = 100;
if (_flags.isTalkie) {
if (_flags.lang == Common::DE_DEU)
chatY = 35;
voiceIndex = 38;
}
seq_playWsaSyncDialogue(27, voiceIndex, 187, chatX, chatY, chatW, wsaObj, 28, 34, x, y);
break;
case 45:
seq_playTalkText(_flags.isTalkie ? 21 : 17);
break;
case 50:
seq_playTalkText(_flags.isTalkie ? 29 : 25);
break;
default:
break;
}
_seqFrameCounter++;
return 0;
}
int KyraEngine_HoF::seq_finaleFheep(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
uint32 endtime = 0;
int chatX = 0;
int chatY = 0;
int chatW = 0;
int chatFirstFrame = 0;
int chatLastFrame = 0;
uint16 voiceIndex = 0;
switch (frm) {
case -2:
_screen->copyPage(12, 2);
_screen->copyPage(2, 0);
_screen->updateScreen();
seq_sequenceCommand(9);
endtime = _system->getMillis() + 480 * _tickLength;
seq_printCreditsString(49, 240, 20, _seqTextColorMap, 252);
seq_printCreditsString(50, 240, 30, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(51, 240, 40, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(52, 240, 50, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(53, 240, 60, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(54, 240, 70, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(55, 240, 80, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(56, 240, 90, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(57, 240, 100, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(58, 240, 110, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(60, 240, 120, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(61, 240, 130, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(62, 240, 140, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(63, 240, 150, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(64, 240, 160, _seqTextColorMap, _seqTextColor[0]);
delay(endtime - _system->getMillis());
_seqEndTime = 0;
break;
case 0:
_seqTextColor[1] = _screen->findLeastDifferentColor(_seqTextColorPresets, _screen->getPalette(0) + 3, 255) & 0xff;
memset(_seqTextColorMap, _seqTextColor[1], 16);
_seqTextColor[0] = _seqTextColorMap[1] = 0xff;
_screen->setTextColorMap(_seqTextColorMap);
break;
case 2:
seq_playTalkText(_flags.isTalkie ? 25 : 21);
if (_flags.lang == Common::FR_FRA) {
chatX = 92;
chatY = 72;
} else {
chatX = (_flags.lang == Common::DE_DEU) ? 90 : 98;
chatY = 84;
}
if (_flags.isTalkie) {
chatFirstFrame = 8;
chatLastFrame = 9;
voiceIndex = 39;
} else {
chatFirstFrame = 2;
chatLastFrame = -8;
}
chatW = 100;
seq_playWsaSyncDialogue(28, voiceIndex, -1, chatX, chatY, chatW, wsaObj, chatFirstFrame, chatLastFrame, x, y);
if (_flags.isTalkie)
_seqWsaCurrentFrame = 4;
break;
case 9:
seq_playTalkText(_flags.isTalkie ? 24 : 20);
_seqFrameDelay = 100;
break;
default:
break;
}
_seqFrameCounter++;
return 0;
}
int KyraEngine_HoF::seq_finaleFarmer(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
uint32 endtime = 0;
int chatX = 0;
int chatY = 0;
int chatW = 0;
uint16 voiceIndex = 0;
switch (frm) {
case -2:
_screen->copyPage(12, 2);
_screen->copyPage(2, 0);
_screen->updateScreen();
seq_sequenceCommand(9);
endtime = _system->getMillis() + 480 * _tickLength;
seq_printCreditsString(45, 240, 40, _seqTextColorMap, 252);
seq_printCreditsString(46, 240, 50, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(47, 240, 60, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(83, 240, 80, _seqTextColorMap, 252);
seq_printCreditsString(48, 240, 90, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(65, 240, 110, _seqTextColorMap, 252);
seq_printCreditsString(66, 240, 120, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(67, 240, 130, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(68, 240, 140, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(69, 240, 150, _seqTextColorMap, _seqTextColor[0]);
if (_flags.platform == Common::kPlatformFMTowns || _flags.platform == Common::kPlatformPC98)
seq_printCreditsString(104, 240, 160, _seqTextColorMap, _seqTextColor[0]);
delay(endtime - _system->getMillis());
_seqEndTime = 0;
break;
case 0:
_seqTextColor[1] = 1 + (_screen->findLeastDifferentColor(_seqTextColorPresets, _screen->getPalette(0) + 3, 254) & 0xff);
memset(_seqTextColorMap, _seqTextColor[1], 16);
_seqTextColorMap[1] = _seqTextColor[0] = 1 + (_screen->findLeastDifferentColor(_seqTextColorPresets + 3, _screen->getPalette(0) + 3, 254) & 0xff);
_screen->setTextColorMap(_seqTextColorMap);
seq_playTalkText(_flags.isTalkie ? 30 : 26);
break;
case 6:
if (_flags.isTalkie)
seq_playTalkText(18);
break;
case 12:
if (!_flags.isTalkie)
seq_playTalkText(14);
chatX = 90;
chatY = 30;
chatW = 100;
if (_flags.isTalkie) {
if (_flags.lang == Common::FR_FRA || _flags.lang == Common::DE_DEU) {
chatX = 75;
chatY = 25;
}
voiceIndex = 40;
}
seq_playWsaSyncDialogue(29, voiceIndex, 150, chatX, chatY, chatW, wsaObj, 12, -21, x, y);
break;
default:
break;
}
_seqFrameCounter++;
return 0;
}
int KyraEngine_HoF::seq_finaleFuards(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
uint32 endtime = 0;
int chatX = 0;
int chatY = 0;
int chatW = 0;
int chatFirstFrame = 0;
int chatLastFrame = 0;
int textCol = 0;
uint16 voiceIndex = 0;
switch (frm) {
case -2:
seq_sequenceCommand(9);
endtime = _system->getMillis() + 480 * _tickLength;
seq_printCreditsString(70, 240, 20, _seqTextColorMap, 252);
seq_printCreditsString(71, 240, 30, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(72, 240, 40, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(73, 240, 50, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(74, 240, 60, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(75, 240, 70, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(101, 240, 80, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(102, 240, 90, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(87, 240, 100, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(88, 240, 110, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(89, 240, 120, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(90, 240, 130, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(91, 240, 140, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(92, 240, 150, _seqTextColorMap, _seqTextColor[0]);
delay(endtime - _system->getMillis());
_seqEndTime = 0;
break;
case 0:
for (int i = 0; i < 0x300; i++)
_screen->getPalette(0)[i] &= 0x3f;
_seqTextColor[1] = 0xCf;
memset(_seqTextColorMap, _seqTextColor[1], 16);
_seqTextColor[0] = _seqTextColorMap[1] = 0xfe;
_screen->setTextColorMap(_seqTextColorMap);
break;
case 6:
_seqFrameDelay = 20;
if (_flags.isTalkie) {
chatX = 82;
textCol = 143;
chatFirstFrame = 16;
chatLastFrame = 21;
voiceIndex = 41;
} else {
chatX = 62;
textCol = 137;
chatFirstFrame = 9;
chatLastFrame = 13;
}
chatY = (_flags.lang == Common::FR_FRA || _flags.lang == Common::DE_DEU) ? 88 :100;
chatW = 80;
seq_playWsaSyncDialogue(30, voiceIndex, 137, chatX, chatY, chatW, wsaObj, chatFirstFrame, chatLastFrame, x, y);
if (_flags.isTalkie)
_seqWsaCurrentFrame = 8;
break;
case 9:
case 16:
if (_flags.isTalkie) {
if (frm == 16)
break;
chatX = 64;
textCol = 137;
chatFirstFrame = 9;
chatLastFrame = 13;
voiceIndex = 42;
} else {
if (frm == 9)
break;
chatX = 80;
textCol = 143;
chatFirstFrame = 16;
chatLastFrame = 21;
}
chatY = 100;
chatW = 100;
seq_playWsaSyncDialogue(31, voiceIndex, 143, chatX, chatY, chatW, wsaObj, chatFirstFrame, chatLastFrame, x, y);
if (_flags.isTalkie)
_seqWsaCurrentFrame = 21;
break;
default:
break;
}
_seqFrameCounter++;
return 0;
}
int KyraEngine_HoF::seq_finaleFirates(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
uint32 endtime = 0;
int chatX = 0;
int chatY = 0;
int chatW = 0;
uint16 voiceIndex = 0;
switch (frm) {
case -2:
_screen->copyPage(12, 2);
_screen->copyPage(2, 0);
_screen->updateScreen();
seq_sequenceCommand(9);
endtime = _system->getMillis() + 480 * _tickLength;
seq_printCreditsString(76, 240, 40, _seqTextColorMap, 252);
seq_printCreditsString(77, 240, 50, _seqTextColorMap, 252);
seq_printCreditsString(78, 240, 60, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(79, 240, 70, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(80, 240, 80, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(84, 240, 100, _seqTextColorMap, 252);
seq_printCreditsString(85, 240, 110, _seqTextColorMap, _seqTextColor[0]);
seq_printCreditsString(99, 240, 130, _seqTextColorMap, 252);
seq_printCreditsString(100, 240, 140, _seqTextColorMap, _seqTextColor[0]);
delay(endtime - _system->getMillis());
_seqEndTime = 0;
break;
case 0:
_seqTextColor[1] = _screen->findLeastDifferentColor(_seqTextColorPresets, _screen->getPalette(0) + 3, 255) & 0xff;
memset(_seqTextColorMap, _seqTextColor[1], 16);
_seqTextColor[0] = _seqTextColorMap[1] = 0xff;
_screen->setTextColorMap(_seqTextColorMap);
break;
case 6:
seq_playTalkText(_flags.isTalkie ? 31 : 27);
break;
case 14:
case 15:
if (!((frm == 15 && !_flags.isTalkie) || (frm == 14 && _flags.isTalkie)))
break;
seq_playTalkText(_flags.isTalkie ? 31 : 27);
if (_flags.lang == Common::DE_DEU) {
chatX = 82;
chatY = 84;
chatW = 140;
} else {
chatX = 74;
chatY = (_flags.lang == Common::FR_FRA) ? 96: 108;
chatW = 80;
}
if (_flags.isTalkie)
voiceIndex = 43;
seq_playWsaSyncDialogue(32, voiceIndex, 137, chatX, chatY, chatW, wsaObj, 14, 16, x, y);
break;
case 28:
seq_playTalkText(_flags.isTalkie ? 32 : 28);
break;
case 29:
seq_playTalkText(_flags.isTalkie ? 33 : 29);
break;
case 31:
if (_flags.isTalkie)
voiceIndex = 44;
chatX = 90;
chatY = (_flags.lang == Common::DE_DEU) ? 60 : 76;
chatW = 80;
seq_playWsaSyncDialogue(33, voiceIndex, 143, chatX, chatY, chatW, wsaObj, 31, 34, x, y);
break;
case 35:
_seqFrameDelay = 300;
break;
default:
break;
}
_seqFrameCounter++;
return 0;
}
int KyraEngine_HoF::seq_finaleFrash(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
int tmp = 0;
switch (frm) {
case -2:
_screen->setCurPage(2);
_screen->clearCurPage();
_screen->copyPage(2, 12);
_screen->copyPage(2, 0);
_screen->updateScreen();
_seqFrameCounter = 0;
seq_loadNestedSequence(0, kSequenceFiggle);
break;
case -1:
if (_flags.isTalkie)
seq_finaleActorScreen();
_seqSpecialFlag = _flags.isTalkie;
break;
case 0:
if (_seqFrameCounter == 1) {
_sound->playTrack(4);
_seqTextColor[1] = _screen->findLeastDifferentColor(_seqTextColorPresets, _screen->getPalette(0) + 3, 255) & 0xff;
memset(_seqTextColorMap, _seqTextColor[1], 16);
_seqTextColor[0] = _seqTextColorMap[1] = 0xff;
_screen->setTextColorMap(_seqTextColorMap);
}
_seqFrameDelay = 10;
break;
case 1:
if (_seqFrameCounter < 20 && _seqSpecialFlag) {
_seqWsaCurrentFrame = 0;
} else {
_seqFrameDelay = _flags.isTalkie ? 500 : (300 + _rnd.getRandomNumberRng(1, 300));
seq_playTalkText(_flags.isTalkie ? 26 : 22);
if (_seqSpecialFlag) {
_seqFrameCounter = 3;
_seqSpecialFlag = false;
}
}
break;
case 2:
_seqFrameDelay = 20;
break;
case 3:
seq_playTalkText(_flags.isTalkie ? 27 : 23);
_seqFrameDelay = _flags.isTalkie ? 500 : (300 + _rnd.getRandomNumberRng(1, 300));
break;
case 4:
_seqFrameDelay = 10;
break;
case 5:
seq_playTalkText(_flags.isTalkie ? 27 : 23);
tmp = _seqFrameCounter / 6;
if (tmp == 2)
_seqFrameDelay = _flags.isTalkie ? 7 : (1 + _rnd.getRandomNumberRng(1, 10));
else if (tmp < 2)
_seqFrameDelay = _flags.isTalkie ? 500 : (300 + _rnd.getRandomNumberRng(1, 300));
break;
case 6:
_seqFrameDelay = 10;
tmp = _seqFrameCounter / 6;
if (tmp == 2)
_seqWsaCurrentFrame = 4;
else if (tmp < 2)
_seqWsaCurrentFrame = 0;
break;
case 7:
_seqFrameCounter = 0;
_seqFrameDelay = 5;
seq_playTalkText(_flags.isTalkie ? 26 : 22);
break;
case 11:
if (_seqFrameCounter < 8)
_seqWsaCurrentFrame = 8;
break;
default:
break;
}
_seqFrameCounter++;
return 0;
}
void KyraEngine_HoF::seq_finaleActorScreen() {
static const uint8 colormap[] = {0, 0, 102, 102, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0};
static const ScreenDim d = { 0x00, 0x0C, 0x28, 0xB4, 0xFF, 0x00, 0x00, 0x00 };
_screen->loadBitmap("finale.cps", 3, 3, _screen->_currentPalette);
_screen->setFont(Screen::FID_GOLDFONT_FNT);
int talkieCreditsSize, talkieCreditsSpecialSize;
const uint8 *talkieCredits = _staticres->loadRawData(k2SeqplayCredits, talkieCreditsSize);
const char *const *talkieCreditsSpecial = _staticres->loadStrings(k2SeqplayCreditsSpecial, talkieCreditsSpecialSize);
_sound->setSoundList(&_soundData[kMusicIngame]);
_sound->loadSoundFile(3);
_sound->playTrack(3);
_screen->setTextColorMap(colormap);
_screen->copyRegion(0, 0, 0, 0, 320, 200, 2, 0);
_screen->updateScreen();
_screen->fadeFromBlack();
_screen->_charWidth = -2;
uint8 *dataPtr = new uint8[0xafd];
memcpy(dataPtr, talkieCredits, talkieCreditsSize);
_staticres->unloadId(k2SeqplayCredits);
seq_displayScrollText(dataPtr, &d, 2, 6, 5, 1, Screen::FID_GOLDFONT_FNT, Screen::FID_GOLDFONT_FNT, 0, talkieCreditsSpecial);
delay(120);
delete[] dataPtr;
_staticres->unloadId(k2SeqplayCreditsSpecial);
_sound->setSoundList(&_soundData[kMusicFinale]);
_sound->loadSoundFile(0);
}
int KyraEngine_HoF::seq_finaleFiggle(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (_seqFrameCounter == 10)
_seqEndTime = 0;
if (_seqFrameCounter == 10 || _seqFrameCounter == 5 || _seqFrameCounter == 7)
seq_playTalkText(_flags.isTalkie ? 45 : 30);
_seqFrameCounter++;
return frm;
}
int KyraEngine_HoF::seq_demoVirgin(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (!frm)
delay(50 * _tickLength);
return 0;
}
int KyraEngine_HoF::seq_demoWestwood(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (!frm)
_sound->playTrack(2);
return 0;
}
int KyraEngine_HoF::seq_demoTitle(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (!frm) {
_sound->playTrack(3);
} else if (frm == 25) {
delay(60 * _tickLength);
_seqEndTime = 0;
seq_sequenceCommand(0);
}
return 0;
}
int KyraEngine_HoF::seq_demoHill(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (!frm) {
_sound->playTrack(4);
} else if (frm == 25) {
seq_loadNestedSequence(0, kSequenceDemoWater);
_seqFrameDelay--;
} else if (frm > 25 && frm < 50) {
if (_seqFrameDelay > 3)
_seqFrameDelay--;
} else if (frm == 95) {
_seqFrameDelay = 70;
} else if (frm == 96) {
_seqFrameDelay = 7;
} else if (frm == 129) {
seq_resetActiveWSA(0);
}
return 0;
}
int KyraEngine_HoF::seq_demoOuthome(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
switch (frm) {
case 12:
seq_playTalkText(4);
break;
case 32:
seq_playTalkText(7);
break;
case 36:
seq_playTalkText(10);
break;
case 57:
seq_playTalkText(9);
break;
case 80:
case 96:
case 149:
_seqFrameDelay = 70;
break;
case 81:
case 97:
_seqFrameDelay = 5;
break;
case 110:
seq_playTalkText(5);
break;
case 137:
seq_playTalkText(6);
break;
}
return 0;
}
int KyraEngine_HoF::seq_demoWharf(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (!_seqFrameCounter)
seq_loadNestedSequence(0, kSequenceDemoWharf2);
switch (frm) {
case 0:
seq_playTalkText(11);
break;
case 5:
if ((_seqFrameCounter / 8) <= 2 || _activeWSA[0].flags != -1)
_seqWsaCurrentFrame = 0;
else
seq_resetActiveWSA(0);
break;
case 6:
seq_resetActiveWSA(0);
break;
case 8:
case 10:
seq_playTalkText(2);
break;
case 13:
seq_playTalkText(7);
break;
case 16:
seq_playTalkText(12);
break;
default:
break;
}
_seqFrameCounter++;
return 0;
}
int KyraEngine_HoF::seq_demoDinob(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (frm == 0) {
if (!(_seqFrameCounter/8)) {
seq_loadNestedSequence(0, kSequenceDemoDinob2);
_seqWsaCurrentFrame = 0;
}
} else if (frm == 3) {
if (_activeWSA[0].flags != -1) {
_seqWsaCurrentFrame = 0;
} else {
seq_resetActiveWSA(0);
_screen->copyPage(2, 12);
}
} else if (frm == 4) {
seq_resetActiveWSA(0);
}
_seqFrameCounter++;
return 0;
}
int KyraEngine_HoF::seq_demoFisher(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (((_system->getMillis() - _seqStartTime) / (5 * _tickLength)) > 0) {
_seqStartTime = _system->getMillis();
if (!_seqFrameCounter) {
seq_loadNestedSequence(0, kSequenceDemoBail);
seq_loadNestedSequence(1, kSequenceDemoDig);
}
if (_seqScrollTextCounter >= 0x18f && !_seqFrameCounter)
return 0;
if (!_seqFrameCounter) {
_screen->loadBitmap("adtext.cps", 4, 4, 0);
_screen->loadBitmap("adtext2.cps", 6, 6, 0);
_screen->copyPageMemory(6, 0, 4, 64000, 1024);
_screen->copyPageMemory(6, 1023, 6, 0, 64000);
_seqScrollTextCounter = 0;
}
seq_scrollPage();
_seqFrameCounter++;
if (_seqFrameCounter < 0x256 || _seqFrameCounter > 0x31c) {
if (_seqFrameCounter < 0x174 || _seqFrameCounter > 0x1d7) {
if (_seqFrameCounter < 0x84 || _seqFrameCounter > 0xe7) {
_seqScrollTextCounter++;
}
}
}
if (_seqFrameCounter > 0x31e) {
seq_resetActiveWSA(0);
seq_resetActiveWSA(1);
_seqEndTime = 0;
_screen->copyPage(2, 12);
}
} else {
seq_scrollPage();
}
return 0;
}
int KyraEngine_HoF::seq_demoWharf2(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (frm == 69)
_seqWsaCurrentFrame = 8;
return frm;
}
int KyraEngine_HoF::seq_demoDinob2(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
switch (frm) {
case 19:
seq_playTalkText(13);
break;
case 54:
seq_playTalkText(15);
break;
case 61:
seq_playTalkText(16);
break;
case 69:
seq_playTalkText(14);
break;
case 77:
seq_playTalkText(13);
break;
case 79:
_seqWsaCurrentFrame = 4;
break;
}
return frm;
}
int KyraEngine_HoF::seq_demoWater(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
if (frm == 1)
seq_playTalkText(11);
return frm;
}
int KyraEngine_HoF::seq_demoBail(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
return frm;
}
int KyraEngine_HoF::seq_demoDig(WSAMovie_v2 *wsaObj, int x, int y, int frm) {
return frm;
}
uint32 KyraEngine_HoF::seq_activeTextsTimeLeft() {
uint32 res = 0;
for (int i = 0; i < 10; i++) {
uint32 chatend = (_activeText[i].duration + _activeText[i].startTime);
uint32 curtime = _system->getMillis();
if (_activeText[i].duration != -1 && chatend > curtime) {
chatend -= curtime;
if (res < chatend)
res = chatend;
}
}
return res;
}
void KyraEngine_HoF::seq_processWSAs() {
for (int i = 0; i < 8; i++) {
if (_activeWSA[i].flags != -1) {
if (seq_processNextSubFrame(i))
seq_resetActiveWSA(i);
}
}
}
void KyraEngine_HoF::seq_processText() {
Screen::FontId curFont = _screen->setFont(Screen::FID_GOLDFONT_FNT);
int curPage = _screen->setCurPage(2);
char outputStr[70];
for (int i = 0; i < 10; i++) {
if (_activeText[i].startTime + _activeText[i].duration > _system->getMillis() && _activeText[i].duration != -1) {
char *srcStr = seq_preprocessString(_sequenceStrings[_activeText[i].strIndex], _activeText[i].width);
int yPos = _activeText[i].y;
while (*srcStr) {
uint32 linePos = 0;
for (; *srcStr; linePos++) {
if (*srcStr == 0x0d) // Carriage return
break;
outputStr[linePos] = *srcStr;
srcStr++;
}
outputStr[linePos] = 0;
if (*srcStr == 0x0d)
srcStr++;
uint8 textColor = (_activeText[i].textcolor >= 0) ? _activeText[i].textcolor : _seqTextColor[0];
_screen->printText(outputStr, _activeText[i].x - (_screen->getTextWidth(outputStr) / 2), yPos, textColor, 0);
yPos += 10;
}
} else {
_activeText[i].duration = -1;
}
}
_screen->setCurPage(curPage);
_screen->setFont(curFont);
}
char *KyraEngine_HoF::seq_preprocessString(const char *srcStr, int width) {
char *dstStr = _seqProcessedString;
int lineStart = 0;
int linePos = 0;
while (*srcStr) {
while (*srcStr && *srcStr != 0x20) // Space
dstStr[lineStart + linePos++] = *srcStr++;
dstStr[lineStart + linePos] = 0;
int len = _screen->getTextWidth(&dstStr[lineStart]);
if (width >= len && *srcStr) {
dstStr[lineStart + linePos++] = *srcStr++;
} else {
dstStr[lineStart + linePos] = 0x0d; // Carriage return
lineStart += linePos + 1;
linePos = 0;
if (*srcStr)
srcStr++;
}
}
dstStr[lineStart + linePos] = 0;
return strlen(_seqProcessedString) ? dstStr : 0;
}
void KyraEngine_HoF::seq_sequenceCommand(int command) {
uint8 pal[768];
for (int i = 0; i < 8; i++)
seq_resetActiveWSA(i);
switch (command) {
case 0:
memset(pal, 0, 0x300);
_screen->fadePalette(pal, 16);
memcpy (_screen->getPalette(0), pal, 0x300);
memcpy (_screen->getPalette(1), pal, 0x300);
break;
case 1:
memset(pal, 0x3F, 0x300);
//////////XXX
//////////Unused anyway (at least by fm-towns intro/outro)
_screen->fadePalette(pal, 16);
memcpy (_screen->getPalette(0), pal, 0x300);
memcpy (_screen->getPalette(1), pal, 0x300);
break;
case 3:
_screen->copyPage(2, 0);
_screen->fadePalette(_screen->getPalette(0), 16);
memcpy (_screen->getPalette(1), _screen->getPalette(0), 0x300);
break;
case 4:
_screen->copyPage(2, 0);
_screen->fadePalette(_screen->getPalette(0), 36);
memcpy (_screen->getPalette(1), _screen->getPalette(0), 0x300);
break;
case 5:
_screen->copyPage(2, 0);
break;
case 6:
// UNUSED
// seq_loadBLD("library.bld");
break;
case 7:
// UNUSED
// seq_loadBLD("marco.bld");
break;
case 8:
memset(pal, 0, 0x300);
_screen->fadePalette(pal, 16);
memcpy (_screen->getPalette(0), pal, 0x300);
memcpy (_screen->getPalette(1), pal, 0x300);
delay(120 * _tickLength);
break;
case 9:
for (int i = 0; i < 0x100; i++) {
int pv = (_screen->getPalette(0)[3 * i] + _screen->getPalette(0)[3 * i + 1] + _screen->getPalette(0)[3 * i + 2]) / 3;
pal[3 * i] = pal[3 * i + 1] = pal[3 * i + 2] = pv & 0xff;
}
//int a = 0x100;
//int d = (0x800 << 5) - 0x100;
//pal[3 * i] = pal[3 * i + 1] = pal[3 * i + 2] = 0x3f;
_screen->fadePalette(pal, 64);
memcpy (_screen->getPalette(0), pal, 0x300);
memcpy (_screen->getPalette(1), pal, 0x300);
break;
default:
break;
}
}
void KyraEngine_HoF::seq_cmpFadeFrame(const char *cmpFile) {
_screen->copyPage(10, 2);
_screen->copyPage(4, 10);
_screen->clearPage(6);
_screen->loadBitmap(cmpFile, 6, 6, 0);
_screen->copyPage(12, 4);
for (int i = 0; i < 3; i++) {
uint32 endtime = _system->getMillis() + 4 * _tickLength;
_screen->cmpFadeFrameStep(4, 320, 200, 0, 0, 2, 320, 200, 0, 0, 320, 200, 6);
_screen->copyRegion(0, 0, 0, 0, 320, 200, 2, 0);
_screen->updateScreen();
delayUntil(endtime);
}
_screen->copyPage(4, 0);
_screen->updateScreen();
_screen->copyPage(4, 2);
_screen->copyPage(4, 6);
_screen->copyPage(10, 4);
}
void KyraEngine_HoF::seq_playTalkText(uint8 chatNum) {
debugC(9, kDebugLevelMain, "KyraEngine_HoF::seq_playTalkText(%i)", chatNum);
assert(chatNum < _sequenceSoundListSize);
if (chatNum < 12 && !_flags.isDemo && textEnabled())
seq_setTextEntry(chatNum, 160, 168, _sequenceStringsDuration[chatNum], 160);
_speechFile = _sequenceSoundList[chatNum];
_sound->voicePlay(_sequenceSoundList[chatNum]);
}
void KyraEngine_HoF::seq_waitForTextsTimeout() {
debugC(9, kDebugLevelMain, "KyraEngine_HoF::seq_waitForTextsTimeout()");
uint32 longest = seq_activeTextsTimeLeft() + _system->getMillis();
uint32 now = _system->getMillis();
if (textEnabled()) {
if (longest > now)
delay(longest - now);
} else if (speechEnabled()) {
while (snd_voiceIsPlaying())
delay(_tickLength);
}
seq_resetAllTextEntries();
}
void KyraEngine_HoF::seq_resetAllTextEntries() {
debugC(9, kDebugLevelMain, "KyraEngine_HoF::seq_resetAllTextEntries()");
for (int i = 0; i < 10; i++)
_activeText[i].duration = -1;
}
int KyraEngine_HoF::seq_setTextEntry(uint16 strIndex, uint16 posX, uint16 posY, int duration, uint16 width) {
debugC(9, kDebugLevelMain, "KyraEngine_HoF::seq_setTextEntry(%i, %i, %i, %i, %i)", strIndex, posX, posY, duration, width);
for (int i = 0; i < 10; i++) {
if (_activeText[i].duration != -1) {
if (i < 9)
continue;
else
return -1;
}
_activeText[i].strIndex = strIndex;
_activeText[i].x = posX;
_activeText[i].y = posY;
_activeText[i].duration = duration * _tickLength;
_activeText[i].width = width;
_activeText[i].startTime = _system->getMillis();
_activeText[i].textcolor = -1;
return i;
}
return -1;
}
void KyraEngine_HoF::seq_loadNestedSequence(int wsaNum, int seqNum) {
debugC(9, kDebugLevelMain, "KyraEngine_HoF::seq_loadNestedSequence(%i, %i)", wsaNum, seqNum);
if (_activeWSA[wsaNum].flags != -1)
return;
NestedSequence s = _sequences->seqn[seqNum];
if (!_activeWSA[wsaNum].movie) {
_activeWSA[wsaNum].movie = new WSAMovie_v2(this, _screen);
assert(_activeWSA[wsaNum].movie);
}
_activeWSA[wsaNum].movie->close();
_activeWSA[wsaNum].movie->open(s.wsaFile, 0, 0);
if (!_activeWSA[wsaNum].movie->opened()) {
delete _activeWSA[wsaNum].movie;
_activeWSA[wsaNum].movie = 0;
return;
}
_activeWSA[wsaNum].endFrame = s.endFrame;
_activeWSA[wsaNum].startFrame = _activeWSA[wsaNum].currentFrame = s.startframe;
_activeWSA[wsaNum].frameDelay = s.frameDelay;
_activeWSA[wsaNum].movie->setX(0);
_activeWSA[wsaNum].movie->setY(0);
_activeWSA[wsaNum].movie->setDrawPage(_screen->_curPage);
_activeWSA[wsaNum].callback = _callbackN[seqNum];
_activeWSA[wsaNum].control = s.wsaControl;
_activeWSA[wsaNum].flags = s.flags | 1;
_activeWSA[wsaNum].x = s.x;
_activeWSA[wsaNum].y = s.y;
_activeWSA[wsaNum].startupCommand = s.startupCommand;
_activeWSA[wsaNum].finalCommand = s.finalCommand;
_activeWSA[wsaNum].lastFrame = 0xffff;
seq_nestedSequenceFrame(s.startupCommand, wsaNum);
if (!s.startupCommand)
seq_processNextSubFrame(wsaNum);
_activeWSA[wsaNum].nextFrame = _system->getMillis();
}
void KyraEngine_HoF::seq_nestedSequenceFrame(int command, int wsaNum) {
int xa = 0, ya = 0;
command--;
if (!_activeWSA[wsaNum].movie || skipFlag() || _quitFlag || _abortIntroFlag)
return;
switch (command) {
case 0:
_activeWSA[wsaNum].movie->setDrawPage(8);
xa = -_activeWSA[wsaNum].movie->xAdd();
ya = -_activeWSA[wsaNum].movie->yAdd();
_activeWSA[wsaNum].movie->setX(xa);
_activeWSA[wsaNum].movie->setY(ya);
_activeWSA[wsaNum].movie->displayFrame(0, 0);
_activeWSA[wsaNum].movie->setX(0);
_activeWSA[wsaNum].movie->setY(0);
seq_animatedSubFrame(8, 2, 7, 8, _activeWSA[wsaNum].movie->xAdd(), _activeWSA[wsaNum].movie->yAdd(),
_activeWSA[wsaNum].movie->width(), _activeWSA[wsaNum].movie->height(), 1, 2);
break;
case 1:
_activeWSA[wsaNum].movie->setDrawPage(8);
xa = -_activeWSA[wsaNum].movie->xAdd();
ya = -_activeWSA[wsaNum].movie->yAdd();
_activeWSA[wsaNum].movie->setX(xa);
_activeWSA[wsaNum].movie->setY(ya);
_activeWSA[wsaNum].movie->displayFrame(0, 0);
_activeWSA[wsaNum].movie->setX(0);
_activeWSA[wsaNum].movie->setY(0);
seq_animatedSubFrame(8, 2, 7, 8, _activeWSA[wsaNum].movie->xAdd(), _activeWSA[wsaNum].movie->yAdd(),
_activeWSA[wsaNum].movie->width(), _activeWSA[wsaNum].movie->height(), 1, 1);
break;
case 2:
seq_waitForTextsTimeout();
_activeWSA[wsaNum].movie->setDrawPage(8);
xa = -_activeWSA[wsaNum].movie->xAdd();
ya = -_activeWSA[wsaNum].movie->yAdd();
_activeWSA[wsaNum].movie->setX(xa);
_activeWSA[wsaNum].movie->setY(ya);
_activeWSA[wsaNum].movie->displayFrame(0x15, 0);
_activeWSA[wsaNum].movie->setX(0);
_activeWSA[wsaNum].movie->setY(0);
seq_animatedSubFrame(8, 2, 7, 8, _activeWSA[wsaNum].movie->xAdd(), _activeWSA[wsaNum].movie->yAdd(),
_activeWSA[wsaNum].movie->width(), _activeWSA[wsaNum].movie->height(), 0, 2);
break;
case 3:
_screen->copyPage(2, 10);
_activeWSA[wsaNum].movie->setDrawPage(2);
_activeWSA[wsaNum].movie->setX(0);
_activeWSA[wsaNum].movie->setY(0);
_activeWSA[wsaNum].movie->displayFrame(0, 0);
_screen->copyPage(2, 12);
seq_cmpFadeFrame("scene2.cmp");
break;
case 4:
_screen->copyPage(2, 10);
_activeWSA[wsaNum].movie->setDrawPage(2);
_activeWSA[wsaNum].movie->setX(0);
_activeWSA[wsaNum].movie->setY(0);
_activeWSA[wsaNum].movie->displayFrame(0, 0);
_screen->copyPage(2, 12);
seq_cmpFadeFrame("scene3.cmp");
break;
default:
break;
}
}
void KyraEngine_HoF::seq_animatedSubFrame(int srcPage, int dstPage, int delaytime, int steps,
int x, int y, int w, int h, int openClose, int directionFlags) {
if (openClose) {
for (int i = 1; i < steps; i++) {
uint32 endtime = _system->getMillis() + delaytime * _tickLength;
int w2 = (((w * 256) / steps) * i) / 256;
int h2 = (((h * 256) / steps) * i) / 256;
int ym = (directionFlags & 2) ? (h - h2) : 0;
int xm = (directionFlags & 1) ? (w - w2) : 0;
_screen->wsaFrameAnimationStep(0, 0, x + xm, y + ym, w, h, w2, h2, srcPage, dstPage, 0);
_screen->copyPage(dstPage, 6);
_screen->copyPage(dstPage, 0);
_screen->updateScreen();
_screen->copyPage(12, dstPage);
delayUntil(endtime);
}
_screen->wsaFrameAnimationStep(0, 0, x, y, w, h, w, h, srcPage, dstPage, 0);
_screen->copyPage(dstPage, 6);
_screen->copyPage(dstPage, 0);
_screen->updateScreen();
} else {
_screen->copyPage(12, dstPage);
for (int i = steps; i; i--) {
uint32 endtime = _system->getMillis() + delaytime * _tickLength;
int w2 = (((w * 256) / steps) * i) / 256;
int h2 = (((h * 256) / steps) * i) / 256;
int ym = (directionFlags & 2) ? (h - h2) : 0;
int xm = (directionFlags & 1) ? (w - w2) : 0;
_screen->wsaFrameAnimationStep(0, 0, x + xm, y + ym, w, h, w2, h2, srcPage, dstPage, 0);
_screen->copyPage(dstPage, 6);
_screen->copyPage(dstPage, 0);
_screen->updateScreen();
_screen->copyPage(12, dstPage);
delayUntil(endtime);
}
}
}
void KyraEngine_HoF::seq_resetActiveWSA(int wsaNum) {
if (_activeWSA[wsaNum].flags == -1)
return;
_activeWSA[wsaNum].flags = -1;
seq_nestedSequenceFrame(_activeWSA[wsaNum].finalCommand, wsaNum);
_activeWSA[wsaNum].movie->close();
}
void KyraEngine_HoF::seq_unloadWSA(int wsaNum) {
if (_activeWSA[wsaNum].movie) {
_activeWSA[wsaNum].movie->close();
delete _activeWSA[wsaNum].movie;
_activeWSA[wsaNum].movie = 0;
}
}
bool KyraEngine_HoF::seq_processNextSubFrame(int wsaNum) {
uint32 currentFrame = _activeWSA[wsaNum].currentFrame;
uint32 currentTime = _system->getMillis();
if (_activeWSA[wsaNum].callback && currentFrame != _activeWSA[wsaNum].lastFrame) {
_activeWSA[wsaNum].lastFrame = currentFrame;
currentFrame = (this->*_activeWSA[wsaNum].callback)(_activeWSA[wsaNum].movie, _activeWSA[wsaNum].x, _activeWSA[wsaNum].y, currentFrame);
}
if (_activeWSA[wsaNum].movie) {
_activeWSA[wsaNum].movie->setDrawPage(2);
_activeWSA[wsaNum].movie->setX(_activeWSA[wsaNum].x);
_activeWSA[wsaNum].movie->setY(_activeWSA[wsaNum].y);
if (_activeWSA[wsaNum].flags & 0x20) {
_activeWSA[wsaNum].movie->displayFrame(_activeWSA[wsaNum].control[currentFrame].index, 0x4000);
_activeWSA[wsaNum].frameDelay = _activeWSA[wsaNum].control[currentFrame].delay;
} else {
_activeWSA[wsaNum].movie->displayFrame(currentFrame % _activeWSA[wsaNum].movie->frames(), 0x4000);
}
}
if (_activeWSA[wsaNum].flags & 0x10) {
currentFrame = (currentTime - _activeWSA[wsaNum].nextFrame) / (_activeWSA[wsaNum].frameDelay * _tickLength);
} else {
if (((int32)(currentTime - _activeWSA[wsaNum].nextFrame) / (int32)(_activeWSA[wsaNum].frameDelay * _tickLength)) > 0) {
currentFrame++;
_activeWSA[wsaNum].nextFrame = currentTime;
}
}
bool res = false;
if (currentFrame >= _activeWSA[wsaNum].endFrame) {
int sw = ((_activeWSA[wsaNum].flags & 0x1e) - 2);
switch (sw) {
case 0:
res = true;
currentFrame = _activeWSA[wsaNum].endFrame;
_screen->copyPage(2, 12);
break;
case 6:
case 8:
currentFrame = _activeWSA[wsaNum].endFrame - 1;
break;
case 2:
case 10:
currentFrame = _activeWSA[wsaNum].startFrame;
break;
default:
currentFrame = _activeWSA[wsaNum].endFrame - 1;
res = true;
break;
}
}
_activeWSA[wsaNum].currentFrame = currentFrame & 0xffff;
return res;
}
void KyraEngine_HoF::seq_printCreditsString(uint16 strIndex, int x, int y, const uint8 *colorMap, uint8 textcolor) {
uint8 colormap[16];
if (skipFlag() || _quitFlag || _abortIntroFlag || _menuChoice)
return;
memset(&_screen->getPalette(0)[0x2fa], 0x3f, 6);
_screen->getPalette(0)[0x2f6] = 0x3f;
_screen->getPalette(0)[0x2f5] = 0x20;
_screen->getPalette(0)[0x2f4] = 0x30;
colormap[0] = colorMap[0];
colormap[1] = 0xfd;
memcpy(&colormap[2], &colorMap[2], 14);
uint8 seqTextColor0 = _seqTextColor[0];
_seqTextColor[0] = 0xfd;
_screen->setTextColorMap(colormap);
seq_resetAllTextEntries();
seq_setTextEntry(strIndex, x, y, 0x80, 0x78);
seq_processText();
_screen->copyPage(2, 0);
_screen->updateScreen();
_screen->getPalette(0)[0x2f7] = _screen->getPalette(0)[textcolor * 3];
_screen->getPalette(0)[0x2f8] = _screen->getPalette(0)[textcolor * 3 + 1];
_screen->getPalette(0)[0x2f9] = _screen->getPalette(0)[textcolor * 3 + 2];
_screen->fadePalette(_screen->getPalette(0), 0x18);
_seqTextColor[0] = textcolor;
_screen->setTextColorMap(colorMap);
seq_resetAllTextEntries();
seq_setTextEntry(strIndex, x, y, 0x80, 0x78);
seq_processText();
_screen->copyPage(2, 0);
_screen->updateScreen();
_screen->getPalette(0)[0x2f7] = _screen->getPalette(0)[0x2f8] = _screen->getPalette(0)[0x2f9] = 0;
_screen->fadePalette(_screen->getPalette(0), 1);
_screen->copyPage(2, 12);
seq_resetAllTextEntries();
_seqTextColor[0] = seqTextColor0;
}
void KyraEngine_HoF::seq_playWsaSyncDialogue(uint16 strIndex, uint16 vocIndex, int textColor, int x, int y, int width, WSAMovie_v2 *wsa, int firstframe, int lastframe, int wsaXpos, int wsaYpos) {
int dur = int(strlen(_sequenceStrings[strIndex])) * (_flags.isTalkie ? 7 : 15);
int entry = textEnabled() ? seq_setTextEntry(strIndex, x, y, dur, width) : strIndex;
_activeText[entry].textcolor = textColor;
uint32 chatTimeout = _system->getMillis() + dur * _tickLength;
int curframe = firstframe;
if (vocIndex && speechEnabled()) {
while (_sound->voiceIsPlaying() && !skipFlag())
delay(4);
seq_playTalkText(vocIndex);
}
while (_system->getMillis() < chatTimeout && !(_abortIntroFlag || skipFlag())) {
if (lastframe < 0) {
int t = ABS(lastframe);
if (t < curframe)
curframe = t;
}
if (ABS(lastframe) < curframe)
curframe = firstframe;
uint32 frameTimeout = _seqEndTime = _system->getMillis() + _seqFrameDelay * _tickLength;
if (wsa) {
wsa->setDrawPage(2);
wsa->setX(wsaXpos);
wsa->setY(wsaYpos);
wsa->displayFrame(curframe % wsa->frames(), 0);
}
_screen->copyPage(2, 12);
seq_processText();
uint32 tm = _system->getMillis();
if (frameTimeout > tm && chatTimeout > tm)
delay(MIN(frameTimeout - tm, chatTimeout - tm));
if (speechEnabled() && !textEnabled() && !snd_voiceIsPlaying())
break;
_screen->copyPage(2, 0);
_screen->updateScreen();
curframe++;
}
if (_abortIntroFlag || skipFlag())
_sound->voiceStop();
if (ABS(lastframe) < curframe)
curframe = ABS(lastframe);
if (curframe == firstframe)
curframe++;
_seqWsaCurrentFrame = curframe;
}
void KyraEngine_HoF::seq_displayScrollText(uint8 *data, const ScreenDim *d, int tempPage1, int tempPage2, int speed,
int step, Screen::FontId fid1, Screen::FontId fid2, const uint8 *shapeData, const char *const *specialData) {
if (!data)
return;
static const char mark[] = { 5, 13, 0 };
_screen->clearPage(tempPage1);
_screen->clearPage(tempPage2);
_screen->copyRegion(d->sx << 3, d->sy, d->sx << 3, d->sy, d->w << 3, d->h, 0, tempPage1);
struct ScrollTextData {
int16 x;
int16 y;
uint8 *text;
byte unk1;
byte height;
byte adjust;
ScrollTextData() {
x = 0; // 0 11
y = 0; // 2 13
text = 0; // 4 15
unk1 = 0; // 8 19
height = 0; // 9 20
adjust = 0; // 10 21
}
};
ScrollTextData *textData = new ScrollTextData[36];
uint8 *ptr = data;
bool loop = true;
int cnt = 0;
while (loop) {
uint32 endTime = _system->getMillis() + speed * _tickLength;
while (cnt < 35 && *ptr) {
uint16 cH;
if (cnt)
cH = textData[cnt].y + textData[cnt].height + (textData[cnt].height >> 3);
else
cH = d->h;
char *str = (char*)ptr;
ptr = (uint8*)strpbrk(str, mark);
if (!ptr)
ptr = (uint8*)strchr(str, 0);
textData[cnt + 1].unk1 = *ptr;
*ptr = 0;
if (textData[cnt + 1].unk1)
ptr++;
if (*str == 3 || *str == 4)
textData[cnt + 1].adjust = *str++;
else
textData[cnt + 1].adjust = 0;
_screen->setFont(fid1);
if (*str == 1) {
_screen->setFont(fid2);
str++;
} else if (*str == 2) {
str++;
}
textData[cnt + 1].height = _screen->getFontHeight();
switch (textData[cnt + 1].adjust) {
case 3:
textData[cnt + 1].x = 157 - _screen->getTextWidth(str);
break;
case 4:
textData[cnt + 1].x = 161;
break;
default:
textData[cnt + 1].x = (((d->w << 3) - _screen->getTextWidth(str)) >> 1) + 1;
break;
}
if (textData[cnt].unk1 == 5)
cH -= (textData[cnt].height + (textData[cnt].height >> 3));
textData[cnt + 1].y = cH;
textData[cnt + 1].text = (uint8*) str;
cnt++;
}
_screen->copyRegion(d->sx << 3, d->sy, d->sx << 3, d->sy, d->w << 3, d->h, tempPage1, tempPage2);
int cnt2 = 0;
bool palCycle = 0;
while (cnt2 < cnt) {
const char *str = (const char*)textData[cnt2 + 1].text;
const char *str2 = str;
int16 cW = textData[cnt2 + 1].x - 10;
int16 cH = textData[cnt2 + 1].y;
int x = (d->sx << 3) + cW;
int y = d->sy + cH;
int col1 = 255;
if (cH < d->h) {
_screen->setCurPage(tempPage2);
_screen->setFont(fid1);
if (textData[cnt2 + 1].height != _screen->getFontHeight())
_screen->setFont(fid2);
if (specialData) {
if (!strcmp(str, specialData[0])) {
col1 = 112;
char cChar[2] = " ";
while (*str2) {
cChar[0] = *str2;
_screen->printText(cChar, x, y, col1++, 0);
x += _screen->getCharWidth(*str2++);
}
palCycle = true;
} else if (!strcmp(str, specialData[1])) {
col1 = 133;
char cChar[2] = " ";
while (*str2) {
cChar[0] = *str2;
_screen->printText(cChar, x, y, col1--, 0);
x += _screen->getCharWidth(*str2++);
}
palCycle = true;
} else {
_screen->printText(str, x, y, col1, 0);
}
} else {
_screen->printText(str, x, y, col1, 0);
}
_screen->setCurPage(0);
}
textData[cnt2 + 1].y -= step;
cnt2++;
}
_screen->copyRegion(d->sx << 3, d->sy, d->sx << 3, d->sy, d->w << 3, d->h, tempPage2, 0);
_screen->updateScreen();
if (textData[1].y < -10) {
textData[1].text += strlen((char*)textData[1].text);
textData[1].text[0] = textData[1].unk1;
cnt--;
memcpy(&textData[1], &textData[2], cnt * sizeof(ScrollTextData));
}
if (palCycle) {
for (int col = 133; col > 112; col--)
memcpy(_screen->_currentPalette + (col * 3), _screen->_currentPalette + ((col - 1) * 3), 3);
memcpy(_screen->_currentPalette + 336, _screen->_currentPalette + 399, 3);
_screen->setScreenPalette(_screen->_currentPalette);
}
delayUntil(endTime);
if ((cnt < 36) && ((d->sy + d->h) > (textData[cnt].y + textData[cnt].height)) && !skipFlag()) {
resetSkipFlag();
delay(_tickLength * 500);
cnt = 0;
}
if (!cnt || skipFlag())
loop = false;
}
_sound->beginFadeOut();
_screen->fadeToBlack();
_abortIntroFlag= false;
resetSkipFlag();
delete[] textData;
}
void KyraEngine_HoF::seq_scrollPage() {
int dstY, dstH, srcH;
static const ScreenDim d = { 0x00, 0x00, 0x28, 0x320, 0xFF, 0xFE, 0x00, 0x00 };
if (_seqScrollTextCounter - 143 < 0) {
dstY = 144 - _seqScrollTextCounter;
dstH = _seqScrollTextCounter;
srcH = 0;
} else {
dstY = 0;
srcH = _seqScrollTextCounter - 144;
dstH = (400 - srcH <= 144) ? 400 - srcH : 144;
}
if (dstH > 0) {
for (int i = 0; i < 4; i++) {
const ItemAnimData_v1 *def = &_demoAnimData[i];
ActiveItemAnim *a = &_activeItemAnim[i];
_screen->fillRect(12, def->y - 8, 28, def->y + 8, 0, 4);
_screen->drawShape(4, getShapePtr(def->itemIndex + def->frames[a->currentFrame]), 12, def->y - 8, 0, 0);
if(_seqFrameCounter % 2 == 0)
a->currentFrame = ++a->currentFrame % 20;
}
_screen->copyRegionEx(4, 0, srcH, 2, 2, dstY + 24, 320, dstH, &d);
}
}
void KyraEngine_HoF::seq_showStarcraftLogo() {
WSAMovie_v2 *ci = new WSAMovie_v2(this, _screen);
assert(ci);
_screen->clearPage(2);
_res->loadPakFile("INTROGEN.PAK");
int endframe = ci->open("ci.wsa", 0, _screen->_currentPalette);
_res->unloadPakFile("INTROGEN.PAK");
if (!ci->opened()) {
delete ci;
return;
}
_screen->hideMouse();
ci->setX(0);
ci->setY(0);
ci->setDrawPage(2);
ci->displayFrame(0, 0);
_screen->copyPage(2, 0);
_screen->fadeFromBlack();
for (int i = 1; i < endframe; i++) {
uint32 endTime = _system->getMillis() + 50;
if (skipFlag())
break;
ci->displayFrame(i, 0);
_screen->copyPage(2, 0);
_screen->updateScreen();
delay(endTime - _system->getMillis());
}
if(!skipFlag()) {
uint32 endTime = _system->getMillis() + 50;
ci->displayFrame(0, 0);
_screen->copyPage(2, 0);
_screen->updateScreen();
delay(endTime - _system->getMillis());
}
_screen->fadeToBlack();
_screen->showMouse();
_eventList.clear();
delete ci;
}
void KyraEngine_HoF::seq_init() {
_seqProcessedString = new char[200];
_seqWsa = new WSAMovie_v2(this, _screen);
_activeWSA = new ActiveWSA[8];
_activeText = new ActiveText[10];
_res->unloadAllPakFiles();
_res->loadPakFile(StaticResource::staticDataFilename());
if (_flags.useInstallerPackage)
_res->loadPakFile("WESTWOOD.001");
_res->loadFileList(_sequencePakList, _sequencePakListSize);
int numShp = -1;
if (_flags.isDemo && !_flags.isTalkie) {
_demoAnimData = _staticres->loadShapeAnimData_v1(k2SeqplayShapeAnimData, _itemAnimDataSize);
uint8 *shp = _res->fileData("icons.shp", 0);
uint32 outsize = READ_LE_UINT16(shp + 4);
_animShapeFiledata = new uint8[outsize];
Screen::decodeFrame4(shp + 10, _animShapeFiledata, outsize);
delete[] shp;
do {
numShp++;
addShapeToPool(_screen->getPtrToShape(_animShapeFiledata, numShp), numShp);
} while (getShapePtr(numShp));
} else {
MainMenu::StaticData data = {
{ _sequenceStrings[97], _sequenceStrings[96], _sequenceStrings[95], _sequenceStrings[98] },
{ 0x01, 0x04, 0x0C, 0x04, 0x00, 0xd7, 0xd6, 0x00, 0x01, 0x02, 0x03 },
{ 0xd8, 0xda, 0xd9, 0xd8 },
0xd7, 0xd6
};
_menu = new MainMenu(this);
_menu->init(data, MainMenu::Animation());
}
}
void KyraEngine_HoF::seq_uninit() {
delete[] _seqProcessedString;
_seqProcessedString = NULL;
delete[] _activeWSA;
_activeWSA = NULL;
delete[] _activeText;
_activeText = NULL;
delete _seqWsa;
_seqWsa = NULL;
delete[] _animShapeFiledata;
_animShapeFiledata = 0;
if (_flags.isDemo && !_flags.isTalkie)
_staticres->unloadId(k2SeqplayShapeAnimData);
delete _menu;
_menu = 0;
}
#pragma mark -
#pragma mark - Ingame sequences
#pragma mark -
void KyraEngine_HoF::seq_makeBookOrCauldronAppear(int type) {
_screen->hideMouse();
showMessage(0, 0xCF);
if (type == 1)
seq_makeBookAppear();
else if (type == 2)
loadInvWsa("CAULDRON.WSA", 1, 6, 0, -2, -2, 1);
_screen->copyRegionToBuffer(2, 0, 0, 320, 200, _screenBuffer);
_screen->loadBitmap("_PLAYALL.CPS", 3, 3, 0);
static const uint8 bookCauldronRects[] = {
0x46, 0x90, 0x7F, 0x2B, // unknown rect (maybe unused?)
0xCE, 0x90, 0x2C, 0x2C, // book rect
0xFA, 0x90, 0x46, 0x2C // cauldron rect
};
int x = bookCauldronRects[type*4+0];
int y = bookCauldronRects[type*4+1];
int w = bookCauldronRects[type*4+2];
int h = bookCauldronRects[type*4+3];
_screen->copyRegion(x, y, x, y, w, h, 2, 0, Screen::CR_NO_P_CHECK);
_screen->copyBlockToPage(2, 0, 0, 320, 200, _screenBuffer);
if (type == 2) {
int32 countdown = _rnd.getRandomNumberRng(45, 80);
_timer->setCountdown(2, countdown * 60);
}
_screen->showMouse();
}
void KyraEngine_HoF::seq_makeBookAppear() {
_screen->hideMouse();
displayInvWsaLastFrame();
showMessage(0, 0xCF);
loadInvWsa("BOOK2.WSA", 0, 4, 2, -1, -1, 0);
uint8 *rect = new uint8[_screen->getRectSize(_invWsa.w, _invWsa.h)];
assert(rect);
_screen->copyRegionToBuffer(_invWsa.page, _invWsa.x, _invWsa.y, _invWsa.w, _invWsa.h, rect);
_invWsa.running = false;
snd_playSoundEffect(0xAF);
_invWsa.wsa->setX(0);
_invWsa.wsa->setY(0);
_invWsa.wsa->setDrawPage(_invWsa.page);
while (true) {
_invWsa.timer = _system->getMillis() + _invWsa.delay * _tickLength;
_screen->copyBlockToPage(_invWsa.page, _invWsa.x, _invWsa.y, _invWsa.w, _invWsa.h, rect);
_invWsa.wsa->displayFrame(_invWsa.curFrame, 0x4000, 0, 0);
if (_invWsa.page)
_screen->copyRegion(_invWsa.x, _invWsa.y, _invWsa.x, _invWsa.y, _invWsa.w, _invWsa.h, _invWsa.page, 0, Screen::CR_NO_P_CHECK);
++_invWsa.curFrame;
if (_invWsa.curFrame >= _invWsa.lastFrame && !_quitFlag)
break;
switch (_invWsa.curFrame) {
case 39:
snd_playSoundEffect(0xCA);
break;
case 50:
snd_playSoundEffect(0x6A);
break;
case 72:
snd_playSoundEffect(0xCB);
break;
case 85:
snd_playSoundEffect(0x38);
break;
default:
break;
}
do {
update();
} while (_invWsa.timer > _system->getMillis() && !skipFlag());
}
closeInvWsa();
delete[] rect;
_invWsa.running = false;
_screen->showMouse();
}
} // end of namespace Kyra