scummvm/engines/kyra/kyra_v3.cpp

1272 lines
31 KiB
C++
Raw Normal View History

/* 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.
2007-04-15 13:44:49 +00:00
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
* You should have received a copy of the GNU General Public License
* 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.h"
#include "kyra/kyra_v3.h"
#include "kyra/screen_v3.h"
2007-04-15 13:44:49 +00:00
#include "kyra/wsamovie.h"
#include "kyra/sound.h"
#include "kyra/text_v3.h"
2007-04-15 13:44:49 +00:00
#include "kyra/vqa.h"
#include "kyra/gui.h"
#include "kyra/timer.h"
2007-04-15 13:44:49 +00:00
#include "common/system.h"
#include "common/config-manager.h"
namespace Kyra {
KyraEngine_v3::KyraEngine_v3(OSystem *system, const GameFlags &flags) : KyraEngine(system, flags) {
2007-04-15 13:44:49 +00:00
_soundDigital = 0;
_musicSoundChannel = -1;
_menuAudioFile = "TITLE1.AUD";
_curMusicTrack = -1;
_itemBuffer1 = _itemBuffer2 = 0;
_scoreFile = 0;
_cCodeFile = 0;
_scenesFile = 0;
_itemFile = 0;
_gamePlayBuffer = 0;
_interface = _interfaceCommandLine = 0;
_costPalBuffer = 0;
_animObjects = 0;
_sceneAnims = 0;
memset(_sceneShapes, 0, sizeof(_sceneShapes));
memset(_sceneAnimMovie, 0, sizeof(_sceneAnimMovie));
_gfxBackUpRect = 0;
_itemList = 0;
_malcolmShapes = 0;
_paletteOverlay = 0;
_sceneList = 0;
memset(&_mainCharacter, 0, sizeof(_mainCharacter));
_mainCharacter.sceneId = 9;
_mainCharacter.unk4 = 0x4C;
_mainCharacter.facing = 5;
_mainCharacter.animFrame = 0x57;
_mainCharacter.walkspeed = 5;
memset(_mainCharacter.inventory, -1, sizeof(_mainCharacter.inventory));
_text = 0;
_commandLineY = 189;
_inventoryState = false;
memset(&_sceneScriptState, 0, sizeof(_sceneScriptState));
memset(&_sceneScriptData, 0, sizeof(_sceneScriptData));
memset(_wsaSlots, 0, sizeof(_wsaSlots));
_updateCharPosNextUpdate = 0;
memset(_characterAnimTable, 0, sizeof(_characterAnimTable));
_overwriteSceneFacing = false;
_maskPageMinY = _maskPageMaxY = 0;
_sceneStrings = 0;
_enterNewSceneLock = 0;
_mainCharX = _mainCharY = -1;
_animList = 0;
_drawNoShapeFlag = false;
_wsaPlayingVQA = false;
_lastCharPalLayer = -1;
_charPalUpdate = false;
_runFlag = false;
_unk5 = 0;
_unkSceneScreenFlag1 = false;
_noScriptEnter = true;
_itemInHand = _handItemSet = -1;
_unk3 = -1;
_unk4 = 0;
_loadingState = false;
_noStartupChat = false;
_lastProcessedSceneScript = 0;
_specialSceneScriptRunFlag = false;
_pathfinderFlag = 0;
2007-04-15 13:44:49 +00:00
}
KyraEngine_v3::~KyraEngine_v3() {
delete _screen;
2007-04-15 13:44:49 +00:00
delete _soundDigital;
2007-04-15 13:44:49 +00:00
delete [] _itemBuffer1;
delete [] _itemBuffer2;
delete [] _scoreFile;
delete [] _cCodeFile;
delete [] _scenesFile;
delete [] _itemFile;
delete [] _gamePlayBuffer;
delete [] _interface;
delete [] _interfaceCommandLine;
delete [] _costPalBuffer;
delete [] _animObjects;
delete [] _sceneAnims;
for (uint i = 0; i < ARRAYSIZE(_sceneShapes); ++i)
delete _sceneShapes[i];
for (uint i = 0; i < ARRAYSIZE(_sceneAnimMovie); ++i)
delete _sceneAnimMovie[i];
delete [] _gfxBackUpRect;
delete [] _itemList;
delete [] _paletteOverlay;
delete [] _sceneList;
for (ShapeMap::iterator i = _gameShapes.begin(); i != _gameShapes.end(); ++i) {
delete i->_value;
i->_value = 0;
}
_gameShapes.clear();
_scriptInterpreter->unloadScript(&_sceneScriptData);
for (int i = 0; i < ARRAYSIZE(_wsaSlots); ++i)
delete _wsaSlots[i];
delete [] _sceneStrings;
2007-04-15 13:44:49 +00:00
}
int KyraEngine_v3::init() {
_screen = new Screen_v3(this, _system);
assert(_screen);
if (!_screen->init())
error("_screen->init() failed");
2007-04-15 13:44:49 +00:00
KyraEngine::init();
_soundDigital = new SoundDigital(this, _mixer);
assert(_soundDigital);
if (!_soundDigital->init())
error("_soundDigital->init() failed");
KyraEngine::_text = _text = new TextDisplayer_v3(this, _screen);
assert(_text);
2007-04-15 13:44:49 +00:00
_screen->loadFont(Screen::FID_6_FNT, "6.FNT");
_screen->loadFont(Screen::FID_8_FNT, "8FAT.FNT");
_screen->loadFont(Screen::FID_BOOKFONT_FNT, "BOOKFONT.FNT");
_screen->setFont(Screen::FID_6_FNT);
2007-04-15 13:44:49 +00:00
_screen->setAnimBlockPtr(3500);
_screen->setScreenDim(0);
_res->loadFileToBuf("PALETTE.COL", _screen->getPalette(0), 768);
_screen->setScreenPalette(_screen->getPalette(0));
2007-04-15 13:44:49 +00:00
return 0;
}
int KyraEngine_v3::go() {
bool running = true;
preinit();
_screen->hideMouse();
initMainMenu();
_screen->clearPage(0);
_screen->clearPage(2);
2007-04-15 13:44:49 +00:00
while (running && !_quitFlag) {
_screen->_curPage = 0;
_screen->clearPage(0);
_screen->setScreenPalette(_screen->getPalette(0));
2007-04-15 13:44:49 +00:00
// XXX
playMenuAudioFile();
2007-04-15 13:44:49 +00:00
for (int i = 0; i < 64 && !_quitFlag; ++i) {
uint32 nextRun = _system->getMillis() + 3 * _tickLength;
_menuAnim->displayFrame(i, 0);
2007-04-15 13:44:49 +00:00
_screen->updateScreen();
delayUntil(nextRun);
}
for (int i = 64; i > 29 && !_quitFlag; --i) {
uint32 nextRun = _system->getMillis() + 3 * _tickLength;
_menuAnim->displayFrame(i, 0);
2007-04-15 13:44:49 +00:00
_screen->updateScreen();
delayUntil(nextRun);
}
switch (_menu->handle(3)) {
2007-04-15 13:44:49 +00:00
case 0:
uninitMainMenu();
startup();
runLoop();
2007-04-15 13:44:49 +00:00
running = false;
break;
2007-04-15 13:44:49 +00:00
case 1:
playVQA("K3INTRO");
_wsaPlayingVQA = false;
_screen->hideMouse();
2007-04-15 13:44:49 +00:00
break;
2007-04-15 13:44:49 +00:00
case 2:
//uninitMainMenu();
2007-04-15 13:44:49 +00:00
//show load dialog
//running = false;
break;
2007-04-15 13:44:49 +00:00
case 3:
default:
2008-04-16 21:13:52 +00:00
fadeOutMusic(60);
_screen->fadeToBlack();
uninitMainMenu();
quitGame();
running = false;
2007-04-15 13:44:49 +00:00
break;
}
}
return 0;
}
void KyraEngine_v3::initMainMenu() {
_menuAnim = new WSAMovieV2(this, _screen);
_menuAnim->open("REVENGE.WSA", 1, _screen->getPalette(0));
_menuAnim->setX(0);
_menuAnim->setY(0);
_menuAnim->setDrawPage(0);
memset(_screen->getPalette(0), 0, 3);
_menu = new MainMenu(this);
MainMenu::StaticData data = {
{ _mainMenuStrings[_lang*4+0], _mainMenuStrings[_lang*4+1], _mainMenuStrings[_lang*4+2], _mainMenuStrings[_lang*4+3] },
{ 0x01, 0x04, 0x0C, 0x04, 0x00, 0x80, 0xFF, 0x00, 0x01, 0x02, 0x03 },
{ 0x16, 0x19, 0x1A, 0x16 },
0x80, 0xFF
};
MainMenu::Animation anim;
anim.anim = _menuAnim;
anim.startFrame = 29;
anim.endFrame = 63;
anim.delay = 2;
_menu->init(data, anim);
}
void KyraEngine_v3::uninitMainMenu() {
delete _menuAnim;
_menuAnim = 0;
delete _menu;
_menu = 0;
}
2007-04-15 13:44:49 +00:00
void KyraEngine_v3::playVQA(const char *name) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::playVQA('%s')", name);
2007-04-15 13:44:49 +00:00
VQAMovie vqa(this, _system);
char filename[20];
int size = 0; // TODO: Movie size is 0, 1 or 2.
snprintf(filename, sizeof(filename), "%s%d.VQA", name, size);
if (vqa.open(filename)) {
_soundDigital->stopAllSounds();
2007-04-15 13:44:49 +00:00
_screen->hideMouse();
memcpy(_screen->getPalette(1), _screen->getPalette(0), 768);
fadeOutMusic(60);
2007-04-15 13:44:49 +00:00
_screen->fadeToBlack();
_screen->clearPage(0);
2007-04-15 13:44:49 +00:00
vqa.setDrawPage(0);
vqa.play();
vqa.close();
_soundDigital->stopAllSounds();
2007-04-15 13:44:49 +00:00
_screen->showMouse();
uint8 pal[768];
memset(pal, 1, sizeof(pal));
2007-04-15 13:44:49 +00:00
_screen->setScreenPalette(pal);
_screen->clearPage(0);
memcpy(_screen->getPalette(0), _screen->getPalette(1), 768);
_wsaPlayingVQA = true;
2007-04-15 13:44:49 +00:00
}
}
#pragma mark -
void KyraEngine_v3::playMenuAudioFile() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::playMenuAudioFile()");
2007-04-15 13:44:49 +00:00
if (_soundDigital->isPlaying(_musicSoundChannel))
return;
2008-02-07 23:27:08 +00:00
Common::SeekableReadStream *stream = _res->getFileStream(_menuAudioFile);
if (stream)
_musicSoundChannel = _soundDigital->playSound(stream, true);
2007-04-15 13:44:49 +00:00
}
void KyraEngine_v3::playMusicTrack(int track, int force) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::playMusicTrack(%d, %d)", track, force);
// XXX byte_3C87C compare
2007-04-15 16:41:20 +00:00
if (_musicSoundChannel != -1 && !_soundDigital->isPlaying(_musicSoundChannel))
2007-04-15 13:44:49 +00:00
force = 1;
2007-04-15 16:41:20 +00:00
else if (_musicSoundChannel == -1)
2007-04-15 13:44:49 +00:00
force = 1;
2007-04-15 13:44:49 +00:00
if (track == _curMusicTrack && !force)
return;
2007-04-15 13:44:49 +00:00
stopMusicTrack();
2007-04-15 13:44:49 +00:00
if (_musicSoundChannel == -1) {
assert(track < _soundListSize && track >= 0);
2008-02-07 23:27:08 +00:00
Common::SeekableReadStream *stream = _res->getFileStream(_soundList[track]);
if (stream)
_musicSoundChannel = _soundDigital->playSound(stream);
2007-04-15 13:44:49 +00:00
}
_curMusicTrack = track;
2007-04-15 13:44:49 +00:00
}
void KyraEngine_v3::stopMusicTrack() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::stopMusicTrack()");
2008-02-07 23:27:08 +00:00
if (_musicSoundChannel != -1 && _soundDigital->isPlaying(_musicSoundChannel))
2007-04-15 13:44:49 +00:00
_soundDigital->stopSound(_musicSoundChannel);
2007-04-15 13:44:49 +00:00
_curMusicTrack = -1;
_musicSoundChannel = -1;
}
int KyraEngine_v3::musicUpdate(int forceRestart) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::musicUpdate(%d)", forceRestart);
static uint32 mTimer = 0;
2007-04-15 13:44:49 +00:00
static uint16 lock = 0;
if (ABS<int>(_system->getMillis() - mTimer) > (int)(0x0F * _tickLength))
mTimer = _system->getMillis();
if (_system->getMillis() < mTimer && !forceRestart)
2007-04-15 13:44:49 +00:00
return 1;
if (!lock) {
lock = 1;
if (_musicSoundChannel >= 0) {
if (!_soundDigital->isPlaying(_musicSoundChannel)) {
if (_curMusicTrack != -1)
playMusicTrack(_curMusicTrack, 1);
}
}
lock = 0;
mTimer = _system->getMillis() + 0x0F * _tickLength;
2007-04-15 13:44:49 +00:00
}
2007-04-15 13:44:49 +00:00
return 1;
}
void KyraEngine_v3::fadeOutMusic(int ticks) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::fadeOutMusic(%d)", ticks);
if (_musicSoundChannel >= 0) {
_fadeOutMusicChannel = _musicSoundChannel;
_soundDigital->beginFadeOut(_musicSoundChannel, ticks);
_curMusicTrack = -1;
}
}
2007-04-15 13:44:49 +00:00
#pragma mark -
void KyraEngine_v3::preinit() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::preinit()");
_unkBuffer1040Bytes = new uint8[1040];
_itemBuffer1 = new uint8[72];
_itemBuffer2 = new uint8[144];
initMouseShapes();
initItems();
_screen->setMouseCursor(0, 0, _gameShapes[0]);
}
void KyraEngine_v3::initMouseShapes() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::initMouseShapes()");
uint8 *data = _res->fileData("MOUSE.SHP", 0);
assert(data);
for (int i = 0; i <= 6; ++i)
_gameShapes[i] = _screen->makeShapeCopy(data, i);
delete [] data;
}
2007-04-15 13:44:49 +00:00
void KyraEngine_v3::startup() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::startup()");
for (int i = 0; i < ARRAYSIZE(_wsaSlots); ++i) {
_wsaSlots[i] = new WSAMovieV2(this, _screen);
assert(_wsaSlots[i]);
}
2007-04-15 13:44:49 +00:00
musicUpdate(0);
memset(_flagsTable, 0, sizeof(_flagsTable));
_gamePlayBuffer = new uint8[64000];
2007-04-15 13:44:49 +00:00
musicUpdate(0);
musicUpdate(0);
_interface = new uint8[17920];
_interfaceCommandLine = new uint8[3840];
2007-04-15 13:44:49 +00:00
_screen->setFont(Screen::FID_6_FNT);
//XXX
musicUpdate(0);
//XXX
_costPalBuffer = new uint8[864];
//XXX
_animObjects = new AnimObj[67];
_sceneAnims = new SceneAnim[16];
2007-04-15 13:44:49 +00:00
musicUpdate(0);
memset(_sceneShapes, 0, sizeof(_sceneShapes));
_screenBuffer = new uint8[64000];
2007-04-15 13:44:49 +00:00
musicUpdate(0);
musicUpdate(0);
if (!loadLanguageFile("ITEMS.", _itemFile))
2007-04-15 13:44:49 +00:00
error("couldn't load ITEMS");
if (!loadLanguageFile("C_CODE.", _cCodeFile))
error("couldn't load C_CODE");
if (!loadLanguageFile("SCENES.", _scenesFile))
2007-04-15 13:44:49 +00:00
error("couldn't load SCENES");
//XXX
if ((_actorFileSize = loadLanguageFile("_ACTOR.", _actorFile)) == 0)
error("couldn't load _ACTOR");
musicUpdate(0);
//XXX
musicUpdate(0);
openTalkFile(0);
musicUpdate(0);
_currentTalkFile = 0;
openTalkFile(1);
//XXX
loadCostPal();
musicUpdate(0);
for (int i = 0; i < 16; ++i) {
_sceneAnims[i].flags = 0;
_sceneAnimMovie[i] = new WSAMovieV2(this, _screen);
assert(_sceneAnimMovie[i]);
}
_screen->_curPage = 0;
//XXX
2007-04-15 13:44:49 +00:00
musicUpdate(0);
updateMalcolmShapes();
_gfxBackUpRect = new uint8[_screen->getRectSize(32, 32)];
_itemList = new Item[50];
resetItemList();
loadShadowShape();
//loadButtonShapes();
musicUpdate(0);
loadExtrasShapes();
musicUpdate(0);
loadMalcolmShapes(_malcolmShapes);
musicUpdate(0);
//initInventoryButtonList(1);
loadInterfaceShapes();
musicUpdate(0);
_res->loadFileToBuf("PALETTE.COL", _screen->getPalette(0), 768);
_paletteOverlay = new uint8[256];
_screen->generateOverlay(_screen->getPalette(0), _paletteOverlay, 0xF0, 0x19);
loadInterface();
musicUpdate(0);
clearAnimObjects();
//XXX
musicUpdate(0);
memset(_hiddenItems, -1, sizeof(_hiddenItems));
//resetNewSceneDlgState();
_sceneList = new SceneDesc[98];
musicUpdate(0);
runStartupScript(1, 0);
//openMoondomtrWsa();
_soundDigital->beginFadeOut(_musicSoundChannel, 60);
2008-04-14 20:42:09 +00:00
delayWithTicks(60);
enterNewScene(_mainCharacter.sceneId, _mainCharacter.facing, 0, 0, 1);
_screen->updateScreen();
musicUpdate(0);
_screen->showMouse();
//XXX
}
void KyraEngine_v3::loadCostPal() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::loadCostPal()");
_costPalBuffer = _res->fileData("_COSTPAL.DAT", 0);
}
void KyraEngine_v3::loadShadowShape() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::loadShadowShape()");
_screen->loadBitmap("SHADOW.CSH", 3, 3, 0);
addShapeToPool(_screen->getCPagePtr(3), 421, 0);
}
void KyraEngine_v3::loadExtrasShapes() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::loadExtrasShapes()");
_screen->loadBitmap("EXTRAS.CSH", 3, 3, 0);
for (int i = 0; i < 20; ++i)
addShapeToPool(_screen->getCPagePtr(3), i+433, i);
addShapeToPool(_screen->getCPagePtr(3), 453, 20);
addShapeToPool(_screen->getCPagePtr(3), 454, 21);
2007-04-15 13:44:49 +00:00
}
void KyraEngine_v3::loadInterfaceShapes() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::loadInterfaceShapes()");
_screen->loadBitmap("INTRFACE.CSH", 3, 3, 0);
for (int i = 422; i <= 432; ++i)
addShapeToPool(_screen->getCPagePtr(3), i, i-422);
}
void KyraEngine_v3::loadInterface() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::loadInterface()");
_screen->loadBitmap("INTRFACE.CPS", 3, 3, 0);
memcpy(_interface, _screen->getCPagePtr(3), 17920);
memcpy(_interfaceCommandLine, _screen->getCPagePtr(3), 3840);
}
2007-04-15 13:44:49 +00:00
void KyraEngine_v3::initItems() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::initItems()");
2007-04-15 13:44:49 +00:00
_screen->loadBitmap("ITEMS.CSH", 3, 3, 0);
for (int i = 248; i <= 319; ++i)
addShapeToPool(_screen->getCPagePtr(3), i, i-248);
2007-04-15 13:44:49 +00:00
_screen->loadBitmap("ITEMS2.CSH", 3, 3, 0);
for (int i = 320; i <= 397; ++i)
addShapeToPool(_screen->getCPagePtr(3), i, i-320);
2007-04-15 13:44:49 +00:00
uint32 size = 0;
uint8 *itemsDat = _res->fileData("_ITEMS.DAT", &size);
assert(size >= 72+144);
memcpy(_itemBuffer1, itemsDat , 72);
memcpy(_itemBuffer2, itemsDat+72, 144);
delete [] itemsDat;
_screen->_curPage = 0;
}
void KyraEngine_v3::runStartupScript(int script, int unk1) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::runStartupScript(%d, %d)", script, unk1);
ScriptState state;
ScriptData data;
memset(&state, 0, sizeof(state));
memset(&data, 0, sizeof(data));
char filename[13];
strcpy(filename, "_START0X.EMC");
filename[7] = (script % 10) + '0';
_scriptInterpreter->loadScript(filename, &data, &_opcodes);
_scriptInterpreter->initScript(&state, &data);
_scriptInterpreter->startScript(&state, 0);
state.regs[6] = unk1;
while (_scriptInterpreter->validScript(&state))
_scriptInterpreter->runScript(&state);
_scriptInterpreter->unloadScript(&data);
}
void KyraEngine_v3::openTalkFile(int file) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::openTalkFile(%d)", file);
char talkFilename[16];
if (file == 0) {
strcpy(talkFilename, "ANYTALK.TLK");
} else {
if (_currentTalkFile > 0)
sprintf(talkFilename, "CH%dTALK.TLK", _currentTalkFile);
_res->unloadPakFile(talkFilename);
sprintf(talkFilename, "CH%dTALK.TLK", file);
}
_currentTalkFile = file;
_res->loadPakFile(talkFilename);
}
#pragma mark -
void KyraEngine_v3::addShapeToPool(const uint8 *data, int realIndex, int shape) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::addShapeToPool(%p, %d, %d)", data, realIndex, shape);
ShapeMap::iterator iter = _gameShapes.find(realIndex);
if (iter != _gameShapes.end()) {
delete [] iter->_value;
iter->_value = 0;
}
_gameShapes[realIndex] = _screen->makeShapeCopy(data, shape);
assert(_gameShapes[realIndex]);
}
void KyraEngine_v3::loadMalcolmShapes(int newShapes) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::loadMalcolmShapes(%d)", newShapes);
static const uint8 numberOffset[] = { 3, 3, 4, 4, 3, 3 };
static const uint8 startShape[] = { 0x32, 0x58, 0x78, 0x98, 0xB8, 0xD8 };
static const uint8 endShape[] = { 0x57, 0x77, 0x97, 0xB7, 0xD7, 0xF7 };
static const char * const filenames[] = {
"MSW##.SHP",
"MTA##.SHP",
"MTFL##.SHP",
"MTFR##.SHP",
"MTL##.SHP",
"MTR#.SHP"
};
for (int i = 50; i <= 247; ++i) {
if (i == 87)
continue;
ShapeMap::iterator iter = _gameShapes.find(i);
if (iter != _gameShapes.end()) {
delete iter->_value;
iter->_value = 0;
}
}
const char lowNum = (newShapes % 10) + '0';
const char highNum = (newShapes / 10) + '0';
for (int i = 0; i < 6; ++i) {
char filename[16];
strcpy(filename, filenames[i]);
filename[numberOffset[i]+0] = highNum;
filename[numberOffset[i]+1] = lowNum;
_res->loadFileToBuf(filename, _screenBuffer, 64000);
for (int j = startShape[i]; j < endShape[i]; ++j) {
if (j == 87)
continue;
addShapeToPool(_screenBuffer, j, j-startShape[i]);
}
}
_malcolmShapes = newShapes;
updateMalcolmShapes();
}
void KyraEngine_v3::updateMalcolmShapes() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::updateMalcolmShapes()");
assert(_malcolmShapes >= 0 && _malcolmShapes < _shapeDescsSize);
_malcolmShapeXOffset = _shapeDescs[_malcolmShapes].xOffset;
_malcolmShapeYOffset = _shapeDescs[_malcolmShapes].yOffset;
_animObjects[0].width = _shapeDescs[_malcolmShapes].width;
_animObjects[0].height = _shapeDescs[_malcolmShapes].height;
}
2007-04-15 13:44:49 +00:00
#pragma mark -
void KyraEngine_v3::showMessage(const char *string, uint8 c0, uint8 c1) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::showMessage('%s', %d, %d)", string, c0, c1);
_shownMessage = string;
_screen->hideMouse();
restoreCommandLine();
_restoreCommandLine = false;
if (string) {
int x = _text->getCenterStringX(string, 0, 320);
int pageBackUp = _screen->_curPage;
_screen->_curPage = 0;
_text->printText(string, x, _commandLineY, c0, c1, 0);
_screen->_curPage = pageBackUp;
_screen->updateScreen();
setCommandLineRestoreTimer(7);
}
_screen->showMouse();
}
void KyraEngine_v3::updateCommandLine() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::updateCommandLine()");
if (_restoreCommandLine) {
restoreCommandLine();
_restoreCommandLine = false;
}
}
void KyraEngine_v3::restoreCommandLine() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::restoreCommandLine()");
int y = _inventoryState ? 144 : 188;
_screen->copyBlockToPage(0, 0, y, 320, 12, _interfaceCommandLine);
}
#pragma mark -
void KyraEngine_v3::moveCharacter(int facing, int x, int y) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::moveCharacter(%d, %d, %d)", facing, x, y);
x &= ~3;
y &= ~1;
_mainCharacter.facing = facing;
_screen->hideMouse();
switch (facing) {
case 0:
while (_mainCharacter.y1 > y)
updateCharPosWithUpdate();
break;
case 2:
while (_mainCharacter.x1 < x)
updateCharPosWithUpdate();
break;
case 4:
while (_mainCharacter.y1 < y)
updateCharPosWithUpdate();
break;
case 6:
while (_mainCharacter.x1 > x)
updateCharPosWithUpdate();
break;
default:
break;
}
_screen->showMouse();
}
void KyraEngine_v3::updateCharPosWithUpdate() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::updateCharPosWithUpdate()");
updateCharPos(0, 0);
update();
}
void KyraEngine_v3::updateCharPos(uint8 *table, int force) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::updateCharPos(%p, %d)", table, force);
if (!force || (_updateCharPosNextUpdate > _system->getMillis()))
return;
_mainCharacter.x1 += _updateCharPosXTable[_mainCharacter.facing];
_mainCharacter.y1 += _updateCharPosYTable[_mainCharacter.facing];
updateCharAnimFrame(0, table);
_updateCharPosNextUpdate = _system->getMillis() + _mainCharacter.walkspeed * _tickLength;
}
void KyraEngine_v3::updateCharAnimFrame(int character, uint8 *table) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::updateCharPos(%d, %p)", character, table);
++_mainCharacter.animFrame;
int facing = _mainCharacter.facing;
if (table) {
if (table[0] != table[-1] && table[1] == table[-1]) {
facing = getOppositeFacingDirection(table[-1]);
table[0] = table[-1];
}
}
if (facing) {
if (facing == 7 || facing == 1) {
if (_characterAnimTable[0] > 2)
facing = 0;
memset(_characterAnimTable, 0, sizeof(_characterAnimTable));
} else if (facing == 4) {
++_characterAnimTable[1];
} else if (facing == 5 || facing == 3) {
if (_characterAnimTable[1] > 2)
facing = 4;
memset(_characterAnimTable, 0, sizeof(_characterAnimTable));
}
} else {
++_characterAnimTable[0];
}
switch (facing) {
case 0:
if (_mainCharacter.animFrame < 79 || _mainCharacter.animFrame > 86)
_mainCharacter.animFrame = 79;
break;
case 1: case 2: case 3:
if (_mainCharacter.animFrame < 71 || _mainCharacter.animFrame > 78)
_mainCharacter.animFrame = 71;
break;
case 4:
if (_mainCharacter.animFrame < 55 || _mainCharacter.animFrame > 62)
_mainCharacter.animFrame = 55;
break;
case 5: case 6: case 7:
if (_mainCharacter.animFrame < 63 || _mainCharacter.animFrame > 70)
_mainCharacter.animFrame = 63;
break;
default:
break;
}
updateCharacterAnim(0);
}
void KyraEngine_v3::updateCharPal(int unk1) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::updateCharPal(%d)", unk1);
int layer = _screen->getLayer(_mainCharacter.x1, _mainCharacter.y1) - 1;
const uint8 *src = _costPalBuffer + _malcolmShapes * 72;
uint8 *dst = _screen->getPalette(0) + 432;
const int8 *sceneDatPal = &_sceneDatPalette[layer * 3];
if (layer != _lastCharPalLayer && unk1) {
for (int i = 0, j = 0; i < 72; ++i) {
uint8 col = *dst;
int8 addCol = *src + *sceneDatPal;
addCol = MAX<int8>(0, MIN<int8>(addCol, 63));
addCol = (col - addCol) >> 1;
*dst -= addCol;
++dst;
++src;
++sceneDatPal;
++j;
if (j > 3) {
sceneDatPal = &_sceneDatPalette[layer * 3];
j = 0;
}
}
_charPalUpdate = true;
_screen->setScreenPalette(_screen->getPalette(0));
_lastCharPalLayer = layer;
} else if (_charPalUpdate || !unk1) {
memcpy(dst, src, 72);
for (int i = 0, j = 0; i < 72; ++i) {
uint8 col = *dst + *sceneDatPal;
*dst = MAX<int8>(0, MIN<int8>(col, 63));
++dst;
++sceneDatPal;
++j;
if (j > 3) {
sceneDatPal = &_sceneDatPalette[layer * 3];
j = 0;
}
}
_screen->setScreenPalette(_screen->getPalette(0));
_charPalUpdate = false;
}
}
#pragma mark -
void KyraEngine_v3::runLoop() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::runLoop()");
_runFlag = true;
while (_runFlag && !_quitFlag) {
//XXX deathHandler
//XXX
int inputFlag = checkInput(0/*_mainButtonList*/);
removeInputTop();
update();
_timer->update();
if (inputFlag == 198 || inputFlag == 199) {
_unk3 = _handItemSet;
Common::Point mouse = getMousePos();
handleInput(mouse.x, mouse.y);
}
_system->delayMillis(10);
}
}
void KyraEngine_v3::handleInput(int x, int y) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::handleInput(%d, %d)", x, y);
}
void KyraEngine_v3::update() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::update()");
updateInput();
musicUpdate(0);
refreshAnimObjectsIfNeed();
musicUpdate(0);
updateMouse();
updateSpecialSceneScripts();
updateCommandLine();
//XXX
musicUpdate(0);
_screen->updateScreen();
}
void KyraEngine_v3::updateMouse() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::updateMouse()");
int shape = 0, offsetX = 0, offsetY = 0;
Common::Point mouse = getMousePos();
bool hasItemCollision = checkItemCollision(mouse.x, mouse.y) != -1;
if (mouse.y > 187) {
bool setItemCursor = false;
if (_handItemSet == -6) {
if (mouse.x < 311)
setItemCursor = true;
} else if (_handItemSet == -5) {
if (mouse.x < _sceneMinX || mouse.x > _sceneMaxX)
setItemCursor = true;
} else if (_handItemSet == -4) {
if (mouse.x > 8)
setItemCursor = true;
}
if (setItemCursor) {
setItemMouseCursor();
return;
}
}
if (_inventoryState) {
if (mouse.y >= 144)
return;
//hideInventory();
}
if (hasItemCollision && _handItemSet < -1 && _itemInHand < 0) {
_handItemSet = -1;
_itemInHand = -1;
_screen->setMouseCursor(0, 0, _gameShapes[0]);
}
int type = 0;
if (mouse.y <= 199) {
if (mouse.x <= 8) {
if (_sceneExit4 != 0xFFFF) {
type = -4;
shape = 4;
offsetX = 0;
offsetY = 0;
}
} else if (mouse.x >= 311) {
if (_sceneExit2 != 0xFFFF) {
type = -6;
shape = 2;
offsetX = 13;
offsetY = 8;
}
} else if (mouse.y >= 171) {
if (_sceneExit3 != 0xFFFF) {
if (mouse.x >= _sceneMinX && mouse.x <= _sceneMaxX) {
type = -5;
shape = 3;
offsetX = 8;
offsetY = 13;
}
}
} else if (mouse.y <= 8) {
if (_sceneExit1 != 0xFFFF) {
type = -7;
shape = 1;
offsetX = 8;
offsetY = 0;
}
}
}
for (int i = 0; i < _specialExitCount; ++i) {
if (checkSpecialSceneExit(i, mouse.x, mouse.y)) {
switch (_specialExitTable[20+i]) {
case 0:
type = -7;
shape = 1;
offsetX = 8;
offsetY = 0;
break;
case 2:
type = -6;
shape = 2;
offsetX = 13;
offsetY = 8;
break;
case 4:
type = -5;
shape = 3;
offsetX = 8;
offsetY = 13;
break;
case 6:
type = -4;
shape = 4;
offsetX = 0;
offsetY = 8;
break;
default:
break;
}
}
}
if (type != 0 && type != _handItemSet && !hasItemCollision) {
_handItemSet = type;
_screen->setMouseCursor(offsetX, offsetY, _gameShapes[shape]);
} else if (type == 0 && _handItemSet != _itemInHand && mouse.x > 8 && mouse.x < 311 && mouse.y < 171 && mouse.y > 8) {
setItemMouseCursor();
} else if (mouse.y > 187 && _handItemSet > -4 && type == 0 && !_inventoryState) {
//showInventory();
}
}
void KyraEngine_v3::delay(uint32 millis, bool doUpdate, bool isMainLoop) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::delay(%d, %d, %d)", millis, doUpdate, isMainLoop);
uint32 endTime = _system->getMillis() + millis;
while (endTime > _system->getMillis()) {
if (doUpdate) {
//XXX
update();
}
_system->delayMillis(10);
}
}
#pragma mark -
void KyraEngine_v3::updateInput() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::updateInput()");
Common::Event event;
while (_eventMan->pollEvent(event)) {
switch (event.type) {
case Common::EVENT_QUIT:
_quitFlag = true;
break;
case Common::EVENT_KEYDOWN:
if (event.kbd.keycode == '.' || event.kbd.keycode == Common::KEYCODE_ESCAPE)
_eventList.push_back(Event(event, true));
else if (event.kbd.keycode == 'q' && event.kbd.flags == Common::KBD_CTRL)
_quitFlag = true;
else
_eventList.push_back(event);
break;
case Common::EVENT_LBUTTONDOWN:
_eventList.push_back(Event(event, true));
break;
case Common::EVENT_LBUTTONUP:
case Common::EVENT_MOUSEMOVE:
_eventList.push_back(event);
break;
default:
break;
}
}
}
int KyraEngine_v3::checkInput(Button *buttonList, bool mainLoop) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::checkInput(%p, %d)", (const void*)buttonList, mainLoop);
updateInput();
int keys = 0;
while (_eventList.size()) {
Common::Event event = *_eventList.begin();
bool breakLoop = false;
switch (event.type) {
case Common::EVENT_KEYDOWN:
/*if (event.kbd.keycode >= '1' && event.kbd.keycode <= '9' &&
(event.kbd.flags == Common::KBD_CTRL || event.kbd.flags == Common::KBD_ALT) && mainLoop) {
const char *saveLoadSlot = getSavegameFilename(9 - (event.kbd.keycode - '0') + 990);
if (event.kbd.flags == Common::KBD_CTRL) {
loadGame(saveLoadSlot);
_eventList.clear();
breakLoop = true;
} else {
char savegameName[14];
sprintf(savegameName, "Quicksave %d", event.kbd.keycode - '0');
saveGame(saveLoadSlot, savegameName);
}
} else if (event.kbd.flags == Common::KBD_CTRL) {
if (event.kbd.keycode == 'd')
_debugger->attach();
}*/
break;
case Common::EVENT_MOUSEMOVE: {
Common::Point pos = getMousePos();
_mouseX = pos.x;
_mouseY = pos.y;
_screen->updateScreen();
} break;
case Common::EVENT_LBUTTONDOWN:
case Common::EVENT_LBUTTONUP: {
Common::Point pos = getMousePos();
_mouseX = pos.x;
_mouseY = pos.y;
keys = event.type == Common::EVENT_LBUTTONDOWN ? 199 : (200 | 0x800);
breakLoop = true;
} break;
default:
break;
}
//if (_debugger->isAttached())
// _debugger->onFrame();
if (breakLoop)
break;
_eventList.erase(_eventList.begin());
}
return /*_gui->processButtonList(buttonList, */keys/* | 0x8000)*/;
}
void KyraEngine_v3::removeInputTop() {
debugC(9, kDebugLevelMain, "KyraEngine_v3::removeInputTop()");
if (!_eventList.empty())
_eventList.erase(_eventList.begin());
}
bool KyraEngine_v3::skipFlag() const {
debugC(9, kDebugLevelMain, "KyraEngine_v3::skipFlag()");
for (Common::List<Event>::const_iterator i = _eventList.begin(); i != _eventList.end(); ++i) {
if (i->causedSkip)
return true;
}
return false;
}
void KyraEngine_v3::resetSkipFlag(bool removeEvent) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::resetSkipFlag(%d)", removeEvent);
for (Common::List<Event>::iterator i = _eventList.begin(); i != _eventList.end(); ++i) {
if (i->causedSkip) {
if (removeEvent)
_eventList.erase(i);
else
i->causedSkip = false;
return;
}
}
}
#pragma mark -
int KyraEngine_v3::getDrawLayer(int x, int y) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::getDrawLayer(%d, %d)", x, y);
int layer = _screen->getLayer(x, y) - 1;
layer = _sceneDatLayerTable[layer];
return MAX(0, MIN(layer, 6));
}
int KyraEngine_v3::getScale(int x, int y) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::getScale(%d, %d)", x, y);
return _scaleTable[_screen->getLayer(x, y) - 1];
}
#pragma mark -
2007-04-15 13:44:49 +00:00
int KyraEngine_v3::getMaxFileSize(const char *file) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::getMaxFileSize(%s)", file);
2007-04-15 13:44:49 +00:00
int size = 0;
char buffer[32];
for (int i = 0; i < _languageExtensionSize; ++i) {
strncpy(buffer, file, 32);
size = MAX<uint32>(size, _res->getFileSize(appendLanguage(buffer, i, sizeof(buffer))));
}
return size + 20;
}
char *KyraEngine_v3::appendLanguage(char *buf, int lang, int bufSize) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::appendLanguage([%p|'%s'], %d, %d)", (const void*)buf, buf, lang, bufSize);
2007-04-15 13:44:49 +00:00
assert(lang < _languageExtensionSize);
int size = strlen(buf) + strlen(_languageExtension[lang]);
if (size > bufSize) {
warning("buffer too small to append language extension");
return 0;
}
char *temp = buf + strlen(buf);
bufSize -= strlen(buf);
strncat(temp, _languageExtension[lang], bufSize);
return buf;
}
int KyraEngine_v3::loadLanguageFile(const char *file, uint8 *&buffer) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::loadLanguageFile('%s', %p)", file, (const void*)buffer);
2007-04-15 13:44:49 +00:00
uint32 size = 0;
char nBuf[32];
strncpy(nBuf, file, 32);
buffer = _res->fileData(appendLanguage(nBuf, _lang, sizeof(nBuf)), &size);
return buffer ? size : 0 ;
2007-04-15 13:44:49 +00:00
}
uint8 *KyraEngine_v3::getTableEntry(uint8 *buffer, int id) {
debugC(9, kDebugLevelMain, "KyraEngine_v3::getTableEntry(%p, %d)", (const void*)buffer, id);
uint16 tableEntries = READ_LE_UINT16(buffer);
const uint16 *indexTable = (const uint16*)(buffer + 2);
const uint16 *offsetTable = indexTable + tableEntries;
int num = 0;
while (id != READ_LE_UINT16(indexTable)) {
++indexTable;
++num;
}
return buffer + READ_LE_UINT16(offsetTable + num);
}
#pragma mark -
Movie *KyraEngine_v3::createWSAMovie() {
WSAMovieV2 *movie = new WSAMovieV2(this, _screen);
assert(movie);
return movie;
}
2007-04-15 13:44:49 +00:00
} // end of namespace Kyra
2007-04-15 16:41:20 +00:00