scummvm/engines/drascula/drascula.cpp
2021-12-26 18:48:43 +01:00

1140 lines
27 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 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/
#include "common/events.h"
#include "common/keyboard.h"
#include "common/file.h"
#include "common/config-manager.h"
#include "common/textconsole.h"
#include "common/translation.h"
#include "backends/audiocd/audiocd.h"
#include "engines/util.h"
#include "drascula/drascula.h"
#include "drascula/console.h"
namespace Drascula {
DrasculaEngine::DrasculaEngine(OSystem *syst, const DrasculaGameDescription *gameDesc) : Engine(syst), _gameDescription(gameDesc) {
_charMap = nullptr;
_itemLocations = nullptr;
_polX = nullptr;
_polY = nullptr;
_verbBarX = nullptr;
_x1d_menu = nullptr;
_y1d_menu = nullptr;
_frameX = nullptr;
_candleX = nullptr;
_candleY = nullptr;
_pianistX = nullptr;
_drunkX = nullptr;
_roomPreUpdates = nullptr;
_roomUpdates = nullptr;
_roomActions = nullptr;
_text = nullptr;
_textd = nullptr;
_textb = nullptr;
_textbj = nullptr;
_texte = nullptr;
_texti = nullptr;
_textl = nullptr;
_textp = nullptr;
_textt = nullptr;
_textvb = nullptr;
_textsys = nullptr;
_texthis = nullptr;
_textverbs = nullptr;
_textmisc = nullptr;
_textd1 = nullptr;
_talkSequences = nullptr;
_currentSaveSlot = 0;
term_int = 0;
currentChapter = 0;
_loadedDifferentChapter = false;
_canSaveLoad = false;
musicStopped = 0;
FrameSSN = 0;
globalSpeed = 0;
LastFrame = 0;
flag_tv = 0;
_charMapSize = 0;
_itemLocationsSize = 0;
_polXSize = 0;
_verbBarXSize = 0;
_x1dMenuSize = 0;
_frameXSize = 0;
_candleXSize = 0;
_pianistXSize = 0;
_drunkXSize = 0;
_roomPreUpdatesSize = 0;
_roomUpdatesSize = 0;
_roomActionsSize = 0;
_talkSequencesSize = 0;
_numLangs = 0;
feetHeight = 0;
lowerLimit = 0;
upperLimit = 0;
trackFinal = 0;
_walkToObject = false;
objExit = 0;
_startTime = 0;
hasAnswer = 0;
savedTime = 0;
breakOut = 0;
vonBraunX = 0;
trackVonBraun = 0;
vonBraunHasMoved = 0;
newHeight = 0;
newWidth = 0;
color_solo = 0;
igorX = 0;
igorY = 0;
trackIgor = 0;
drasculaX = 0;
drasculaY = 0;
trackDrascula = 0;
_roomNumber = 0;
numRoomObjs = 0;
takeObject = 0;
pickedObject = 0;
_subtitlesDisabled = 0;
_menuBar = 0;
_menuScreen = 0;
_hasName = 0;
curExcuseLook = 0;
curExcuseAction = 0;
frame_y = 0;
curX = 0;
curY = 0;
_characterMoved = false;
curDirection = 0;
trackProtagonist = 0;
_characterFrame = 0;
_characterVisible = false;
roomX = 0;
roomY = 0;
checkFlags = 0;
doBreak = 0;
stepX = 0;
stepY = 0;
curHeight = 0;
curWidth = 0;
_color = 0;
blinking = 0;
_mouseX = 0;
_mouseY = 0;
_leftMouseButton = 0;
_rightMouseButton = 0;
*textName = 0;
crosshairCursor = nullptr;
mouseCursor = nullptr;
bgSurface = nullptr;
backSurface = nullptr;
cursorSurface = nullptr;
drawSurface3 = nullptr;
drawSurface2 = nullptr;
tableSurface = nullptr;
extraSurface = nullptr;
screenSurface = nullptr;
frontSurface = nullptr;
previousMusic = 0;
roomMusic = 0;
_rnd = new Common::RandomSource("drascula");
const Common::FSNode gameDataDir(ConfMan.get("path"));
SearchMan.addSubDirectoryMatching(gameDataDir, "audio");
_system->getAudioCDManager()->open();
_lang = kEnglish;
_keyBufferHead = _keyBufferTail = 0;
_roomHandlers = nullptr;
}
DrasculaEngine::~DrasculaEngine() {
delete _rnd;
stopSound();
freeRoomsTable();
free(_charMap);
free(_itemLocations);
free(_polX);
free(_polY);
free(_verbBarX);
free(_x1d_menu);
free(_y1d_menu);
free(_frameX);
free(_candleX);
free(_candleY);
free(_pianistX);
free(_drunkX);
free(_roomPreUpdates);
free(_roomUpdates);
free(_roomActions);
free(_talkSequences);
freeTexts(_text);
freeTexts(_textd);
freeTexts(_textb);
freeTexts(_textbj);
freeTexts(_texte);
freeTexts(_texti);
freeTexts(_textl);
freeTexts(_textp);
freeTexts(_textt);
freeTexts(_textvb);
freeTexts(_textsys);
freeTexts(_texthis);
freeTexts(_textverbs);
freeTexts(_textmisc);
freeTexts(_textd1);
}
bool DrasculaEngine::hasFeature(EngineFeature f) const {
return
(f == kSupportsReturnToLauncher || f == kSupportsLoadingDuringRuntime || f == kSupportsSavingDuringRuntime);
}
Common::Error DrasculaEngine::run() {
// Initialize backend
initGraphics(320, 200);
switch (getLanguage()) {
case Common::EN_ANY:
_lang = kEnglish;
break;
case Common::ES_ESP:
_lang = kSpanish;
break;
case Common::DE_DEU:
_lang = kGerman;
break;
case Common::FR_FRA:
_lang = kFrench;
break;
case Common::IT_ITA:
_lang = kItalian;
break;
case Common::RU_RUS:
_lang = kRussian;
break;
default:
warning("Unknown game language. Falling back to English");
_lang = kEnglish;
}
setDebugger(new Console(this));
if (!loadDrasculaDat())
return Common::kUnknownError;
checkForOldSaveGames();
setupRoomsTable();
loadArchives();
// Setup mixer
syncSoundSettings();
currentChapter = 1; // values from 1 to 6 will start each part of game
_loadedDifferentChapter = false;
setTotalPlayTime(0);
// Check if a save is loaded from the launcher
int directSaveSlotLoading = ConfMan.getInt("save_slot");
if (directSaveSlotLoading >= 0) {
// Set the current chapter to -1. This forces the load to happen
// later during the game loop, and not now.
currentChapter = -1;
loadGame(directSaveSlotLoading);
currentChapter++;
}
if (!existExtractedCDAudioFiles()
&& !isDataAndCDAudioReadFromSameCD()) {
warnMissingExtractedCDAudio();
}
allocMemory();
while (!shouldQuit()) {
int i;
takeObject = 0;
_menuBar = false;
_menuScreen = false;
_hasName = false;
frame_y = 0;
curX = -1;
_characterMoved = false;
trackProtagonist = 3;
_characterFrame = 0;
_characterVisible = true;
checkFlags = 1;
doBreak = 0;
_walkToObject = false;
stepX = STEP_X;
stepY = STEP_Y;
curHeight = CHARACTER_HEIGHT;
curWidth = CHARACTER_WIDTH;
feetHeight = FEET_HEIGHT;
hasAnswer = 0;
savedTime = 0;
breakOut = 0;
vonBraunX = 120;
trackVonBraun = 1;
vonBraunHasMoved = 0;
term_int = 0;
musicStopped = 0;
globalSpeed = 0;
curExcuseLook = 0;
curExcuseAction = 0;
_roomNumber = 0;
for (i = 0; i < 8; i++)
actorFrames[i] = 0;
actorFrames[kFrameVonBraun] = 1;
_subtitlesDisabled = !ConfMan.getBool("subtitles");
if (currentChapter != 3)
loadPic(96, frontSurface, COMPLETE_PAL);
loadPic(99, cursorSurface);
if (currentChapter == 1) {
} else if (currentChapter == 2) {
loadPic("pts.alg", drawSurface2);
} else if (currentChapter == 3) {
loadPic("aux13.alg", bgSurface, COMPLETE_PAL);
loadPic(96, frontSurface);
} else if (currentChapter == 4) {
if (!_loadedDifferentChapter)
animation_castle();
loadPic(96, frontSurface);
clearRoom();
} else if (currentChapter == 5) {
} else if (currentChapter == 6) {
igorX = 105;
igorY = 85;
trackIgor = 1;
drasculaX = 62;
drasculaY = 99;
trackDrascula = 1;
actorFrames[kFramePendulum] = 0;
flag_tv = 0;
}
loadPic(95, tableSurface);
for (i = 0; i < 25; i++)
memcpy(crosshairCursor + i * 40, tableSurface + 225 + (56 + i) * 320, 40);
if (_lang == kSpanish && currentChapter != 6)
loadPic(974, tableSurface);
if (currentChapter != 2) {
loadPic(99, cursorSurface);
loadPic(99, backSurface);
loadPic(97, extraSurface);
}
memset(iconName, 0, sizeof(iconName));
for (i = 0; i < 6; i++)
Common::strlcpy(iconName[i + 1], _textverbs[i], 13);
assignPalette(defaultPalette);
if (!runCurrentChapter()) {
endChapter();
break;
}
endChapter();
if (currentChapter == 6)
break;
currentChapter++;
}
freeMemory();
return Common::kNoError;
}
void DrasculaEngine::endChapter() {
stopSound();
clearRoom();
black();
MusicFadeout();
stopMusic();
}
bool DrasculaEngine::runCurrentChapter() {
int n;
int framesWithoutAction = 0;
_rightMouseButton = 0;
previousMusic = -1;
if (currentChapter != 2) {
int soc = 0;
for (n = 0; n < 6; n++) {
soc = soc + CHARACTER_WIDTH;
_frameX[n] = soc;
}
}
for (n = 1; n < ARRAYSIZE(inventoryObjects); n++)
inventoryObjects[n] = 0;
for (n = 0; n < NUM_FLAGS; n++)
flags[n] = 0;
if (currentChapter == 2) {
flags[16] = 1;
flags[17] = 1;
flags[27] = 1;
}
for (n = 1; n < 7; n++)
inventoryObjects[n] = n;
if (currentChapter == 1) {
pickObject(28);
if (!_loadedDifferentChapter)
animation_1_1();
selectVerb(kVerbNone);
loadPic("2aux62.alg", drawSurface2);
trackProtagonist = 1;
objExit = 104;
if (_loadedDifferentChapter) {
if (!loadGame(_currentSaveSlot)) {
return true;
}
} else {
enterRoom(62);
curX = -20;
curY = 56;
walkToPoint(Common::Point(65, 145));
}
// REMINDER: This is a good place to debug animations
} else if (currentChapter == 2) {
addObject(kItemPhone);
trackProtagonist = 3;
objExit = 162;
if (!_loadedDifferentChapter)
enterRoom(14);
else {
if (!loadGame(_currentSaveSlot)) {
return true;
}
}
} else if (currentChapter == 3) {
addObject(kItemPhone);
addObject(kItemEarplugs);
addObject(kItemSickle);
addObject(kItemHandbag);
addObject(kItemCross);
addObject(kItemReefer);
addObject(kItemOneCoin);
flags[1] = 1;
trackProtagonist = 1;
objExit = 99;
if (!_loadedDifferentChapter)
enterRoom(20);
else {
if (!loadGame(_currentSaveSlot)) {
return true;
}
}
// From here onwards the items have different IDs
} else if (currentChapter == 4) {
addObject(kItemPhone2);
addObject(kItemCross2);
addObject(kItemReefer2);
addObject(kItemOneCoin2);
objExit = 100;
if (!_loadedDifferentChapter) {
enterRoom(21);
trackProtagonist = 0;
curX = 235;
curY = 164;
} else {
if (!loadGame(_currentSaveSlot)) {
return true;
}
}
} else if (currentChapter == 5) {
addObject(28);
addObject(7);
addObject(9);
addObject(11);
addObject(13);
addObject(14);
addObject(15);
addObject(17);
addObject(20);
trackProtagonist = 1;
objExit = 100;
if (!_loadedDifferentChapter) {
enterRoom(45);
} else {
if (!loadGame(_currentSaveSlot)) {
return true;
}
}
} else if (currentChapter == 6) {
addObject(28);
addObject(9);
trackProtagonist = 1;
objExit = 104;
if (!_loadedDifferentChapter) {
enterRoom(58);
animation_1_6();
} else {
if (!loadGame(_currentSaveSlot)) {
return true;
}
loadPic("auxdr.alg", drawSurface2);
}
}
showCursor();
while (!shouldQuit()) {
if (!_characterMoved) {
stepX = STEP_X;
stepY = STEP_Y;
}
if (!_characterMoved && _walkToObject) {
trackProtagonist = trackFinal;
_walkToObject = false;
}
if (currentChapter == 2) {
// NOTE: the checks for room number 14 below are a hack used in the original
// game, and move the character to a place where his feet are not drawn above
// the pianist's head. Originally, walkToObject was not updated properly, which
// lead to an incorrect setting of the protagonist's tracking flag (above). This
// made the character start walking off screen, as his actual position was
// different than the displayed one
if (_roomNumber == 3 && (curX == 279) && (curY + curHeight == 101)) {
walkToPoint(Common::Point(178, 121));
walkToPoint(Common::Point(169, 135));
} else if (_roomNumber == 14 && (curX == 214) && (curY + curHeight == 121)) {
_walkToObject = true;
walkToPoint(Common::Point(190, 130));
} else if (_roomNumber == 14 && (curX == 246) && (curY + curHeight == 112)) {
_walkToObject = true;
walkToPoint(Common::Point(190, 130));
}
}
moveCursor();
updateScreen();
if (currentChapter == 2) {
if (musicStatus() == 0 && roomMusic != 0)
playMusic(roomMusic);
} else {
if (musicStatus() == 0)
playMusic(roomMusic);
}
_canSaveLoad = true;
delay(25);
updateEvents();
_canSaveLoad = false;
if (_loadedDifferentChapter)
return true;
if (!_menuScreen && takeObject == 1)
checkObjects();
if (_rightMouseButton == 1 && _menuScreen) {
_rightMouseButton = 0;
if (currentChapter == 2) {
loadPic(menuBackground, cursorSurface);
loadPic(menuBackground, backSurface);
} else {
loadPic(99, cursorSurface);
loadPic(99, backSurface);
}
setPalette((byte *)&gamePalette);
_menuScreen = false;
// FIXME: This call here is in hope that it will catch the rightmouseup event so the
// next if block won't be executed. This too is not a good coding practice. I've recoded it
// with a mutual exclusive if block for the menu. I would commit this properly but I cannot test
// for other (see Desktop) ports right now.
updateEvents();
}
// Do not show the inventory screen in chapter 5, if the right mouse button is clicked
// while the plug (object 16) is held
// Fixes bug #3885 - "DRASCULA: Plug bug"
if (_rightMouseButton == 1 && !_menuScreen &&
!(currentChapter == 5 && pickedObject == 16)) {
_rightMouseButton = 0;
_characterMoved = false;
if (trackProtagonist == 2)
trackProtagonist = 1;
if (currentChapter == 4) {
loadPic("icons2.alg", backSurface);
loadPic("icons2.alg", cursorSurface);
} else if (currentChapter == 5) {
loadPic("icons3.alg", backSurface);
loadPic("icons3.alg", cursorSurface);
} else if (currentChapter == 6) {
loadPic("iconsp.alg", backSurface);
loadPic("iconsp.alg", cursorSurface);
} else {
loadPic("icons.alg", backSurface);
loadPic("icons.alg", cursorSurface);
}
_menuScreen = true;
updateEvents();
selectVerb(kVerbNone);
}
if (_leftMouseButton == 1 && _menuBar) {
selectVerbFromBar();
} else if (_leftMouseButton == 1 && takeObject == 0) {
if (verify1())
return true;
delay(100);
} else if (_leftMouseButton == 1 && takeObject == 1) {
if (verify2())
return true;
}
_menuBar = (_mouseY < 24 && !_menuScreen) ? true : false;
_canSaveLoad = true;
Common::KeyCode key = getScan();
_canSaveLoad = false;
if (_loadedDifferentChapter)
return true;
if (key == Common::KEYCODE_F1 && !_menuScreen) {
selectVerb(kVerbLook);
} else if (key == Common::KEYCODE_F2 && !_menuScreen) {
selectVerb(kVerbPick);
} else if (key == Common::KEYCODE_F3 && !_menuScreen) {
selectVerb(kVerbOpen);
} else if (key == Common::KEYCODE_F4 && !_menuScreen) {
selectVerb(kVerbClose);
} else if (key == Common::KEYCODE_F5 && !_menuScreen) {
selectVerb(kVerbTalk);
} else if (key == Common::KEYCODE_F6 && !_menuScreen) {
selectVerb(kVerbMove);
} else if (key == Common::KEYCODE_F7) {
// ScummVM load screen
if (!scummVMSaveLoadDialog(false))
return true;
} else if (key == Common::KEYCODE_F8) {
selectVerb(kVerbNone);
} else if (key == Common::KEYCODE_F9) {
volumeControls();
} else if (key == Common::KEYCODE_F10) {
if (!ConfMan.getBool("originalsaveload")) {
// ScummVM save screen
scummVMSaveLoadDialog(true);
} else {
// Original save/load screen
if (!saveLoadScreen())
return true;
}
} else if (key == Common::KEYCODE_v) {
_subtitlesDisabled = true;
ConfMan.setBool("subtitles", !_subtitlesDisabled);
print_abc(_textsys[2], 96, 86);
updateScreen();
delay(1410);
} else if (key == Common::KEYCODE_t) {
_subtitlesDisabled = false;
ConfMan.setBool("subtitles", !_subtitlesDisabled);
print_abc(_textsys[3], 94, 86);
updateScreen();
delay(1460);
} else if (key == Common::KEYCODE_ESCAPE) {
if (!confirmExit())
return false;
} else if (currentChapter == 6 && key == Common::KEYCODE_0 && _roomNumber == 61) {
loadPic("alcbar.alg", bgSurface, 255);
}
if (_leftMouseButton != 0 || _rightMouseButton != 0 || key != 0)
framesWithoutAction = 0;
if (framesWithoutAction == 15000) {
screenSaver();
framesWithoutAction = 0;
}
framesWithoutAction++;
}
return false;
}
bool DrasculaEngine::verify1() {
int l;
if (_menuScreen)
removeObject();
else {
for (l = 0; l < numRoomObjs; l++) {
if (_objectRect[l].contains(Common::Point(_mouseX, _mouseY)) && doBreak == 0) {
if (exitRoom(l))
return true;
if (doBreak == 1)
break;
}
}
if (_mouseX > curX && _mouseY > curY
&& _mouseX < curX + curWidth && _mouseY < curY + curHeight)
doBreak = 1;
for (l = 0; l < numRoomObjs; l++) {
if (_objectRect[l].contains(Common::Point(_mouseX, _mouseY)) && doBreak == 0) {
roomX = _roomObject[l].x;
roomY = _roomObject[l].y;
trackFinal = trackObj[l];
doBreak = 1;
_walkToObject = true;
startWalking();
}
}
if (doBreak == 0) {
roomX = CLIP<int16>(_mouseX, _walkRect.left, _walkRect.right);
roomY = CLIP<int16>(_mouseY, _walkRect.top + feetHeight, _walkRect.bottom);
startWalking();
}
doBreak = 0;
}
return false;
}
bool DrasculaEngine::verify2() {
int l;
if (_menuScreen) {
if (pickupObject())
return true;
} else {
if (!strcmp(textName, _textmisc[3]) && _hasName) {
if (checkAction(50))
return true;
} else {
for (l = 0; l < numRoomObjs; l++) {
if (_objectRect[l].contains(Common::Point(_mouseX, _mouseY)) && visible[l] == 1) {
trackFinal = trackObj[l];
_walkToObject = true;
walkToPoint(_roomObject[l]);
if (checkAction(objectNum[l]))
return true;
if (currentChapter == 4)
break;
}
}
}
}
return false;
}
Common::KeyCode DrasculaEngine::getScan() {
updateEvents();
if (_keyBufferHead == _keyBufferTail)
return Common::KEYCODE_INVALID;
Common::KeyCode key = _keyBuffer[_keyBufferTail].keycode;
_keyBufferTail = (_keyBufferTail + 1) % KEYBUFSIZE;
return key;
}
void DrasculaEngine::addKeyToBuffer(Common::KeyState& key) {
if ((_keyBufferHead + 1) % KEYBUFSIZE == _keyBufferTail) {
warning("key buffer overflow");
return;
}
_keyBuffer[_keyBufferHead] = key;
_keyBufferHead = (_keyBufferHead + 1) % KEYBUFSIZE;
}
void DrasculaEngine::flushKeyBuffer() {
updateEvents();
_keyBufferHead = _keyBufferTail = 0;
}
void DrasculaEngine::updateEvents() {
Common::Event event;
Common::EventManager *eventMan = _system->getEventManager();
updateMusic();
while (eventMan->pollEvent(event)) {
switch (event.type) {
case Common::EVENT_KEYDOWN:
addKeyToBuffer(event.kbd);
break;
case Common::EVENT_KEYUP:
break;
case Common::EVENT_MOUSEMOVE:
_mouseX = event.mouse.x;
_mouseY = event.mouse.y;
break;
case Common::EVENT_LBUTTONDOWN:
_leftMouseButton = 1;
break;
case Common::EVENT_LBUTTONUP:
_leftMouseButton = 0;
break;
case Common::EVENT_RBUTTONDOWN:
// We changed semantic and react only on button up event
break;
case Common::EVENT_RBUTTONUP:
_rightMouseButton = 1;
break;
default:
break;
}
}
}
void DrasculaEngine::delay(int ms) {
uint32 end = _system->getMillis() + ms * 2; // originally was 1
do {
_system->delayMillis(10);
updateEvents();
_system->updateScreen();
} while (_system->getMillis() < end && !shouldQuit() && !_loadedDifferentChapter);
}
void DrasculaEngine::pause(int duration) {
delay(duration * 15);
}
int DrasculaEngine::getTime() {
return _system->getMillis() / 10;
}
void DrasculaEngine::reduce_hare_chico(int xx1, int yy1, int xx2, int yy2, int width, int height, int factor, byte *dir_inicio, byte *dir_fin) {
float totalX, totalY;
int n, m;
float pixelX, pixelY;
newWidth = (width * factor) / 100;
newHeight = (height * factor) / 100;
totalX = width / newWidth;
totalY = height / newHeight;
pixelX = xx1;
pixelY = yy1;
for (n = 0; n < newHeight; n++) {
for (m = 0; m < newWidth; m++) {
copyRect((int)pixelX, (int)pixelY, xx2 + m, yy2 + n,
1, 1, dir_inicio, dir_fin);
pixelX += totalX;
}
pixelX = xx1;
pixelY += totalY;
}
}
void DrasculaEngine::hipo_sin_nadie(int counter){
int y = 0, trackCharacter = 0;
if (currentChapter == 3)
y = -1;
do {
counter--;
copyBackground();
if (currentChapter == 3)
updateScreen(0, 0, 0, y, 320, 200, screenSurface);
else
updateScreen(0, 1, 0, y, 320, 198, screenSurface);
if (trackCharacter == 0)
y++;
else
y--;
if (currentChapter == 3) {
if (y == 1)
trackCharacter = 1;
if (y == -1)
trackCharacter = 0;
} else {
if (y == 2)
trackCharacter = 1;
if (y == 0)
trackCharacter = 0;
}
} while (counter > 0);
copyBackground();
updateScreen();
}
bool DrasculaEngine::loadDrasculaDat() {
Common::File in;
Common::String filename = "drascula.dat";
int i;
in.open(filename.c_str());
if (!in.isOpen()) {
const char *msg = _s("Unable to locate the '%s' engine data file.");
Common::U32String errorMessage = Common::U32String::format(_(msg), filename.c_str());
GUIErrorMessage(errorMessage);
warning(msg, filename.c_str());
return false;
}
char buf[256];
int ver;
in.read(buf, 8);
buf[8] = '\0';
if (strcmp(buf, "DRASCULA") != 0) {
const char *msg = _s("The '%s' engine data file is corrupt.");
Common::U32String errorMessage = Common::U32String::format(_(msg), filename.c_str());
GUIErrorMessage(errorMessage);
warning(msg, filename.c_str());
return false;
}
ver = in.readByte();
if (ver != DRASCULA_DAT_VER) {
const char *msg = _s("Incorrect version of the '%s' engine data file found. Expected %d.%d but got %d.%d.");
Common::U32String errorMessage = Common::U32String::format(_(msg), filename.c_str(), DRASCULA_DAT_VER, 0, ver, 0);
GUIErrorMessage(errorMessage);
warning(msg, filename.c_str(), DRASCULA_DAT_VER, 0, ver, 0);
return false;
}
_charMapSize = in.readUint16BE();
_charMap = (CharInfo *)malloc(sizeof(CharInfo) * _charMapSize);
for (i = 0; i < _charMapSize; i++) {
_charMap[i].inChar = in.readByte();
_charMap[i].mappedChar = in.readSint16BE();
_charMap[i].charType = in.readByte();
}
_itemLocationsSize = in.readUint16BE();
_itemLocations = (ItemLocation *)malloc(sizeof(ItemLocation) * _itemLocationsSize);
for (i = 0; i < _itemLocationsSize; i++) {
_itemLocations[i].x = in.readSint16BE();
_itemLocations[i].y = in.readSint16BE();
}
_polXSize = in.readUint16BE();
_polX = (int *)malloc(sizeof(int) * _polXSize);
_polY = (int *)malloc(sizeof(int) * _polXSize);
for (i = 0; i < _polXSize; i++) {
_polX[i] = in.readSint16BE();
_polY[i] = in.readSint16BE();
}
_verbBarXSize = in.readUint16BE();
_verbBarX = (int *)malloc(sizeof(int) * _verbBarXSize);
for (i = 0; i < _verbBarXSize; i++) {
_verbBarX[i] = in.readSint16BE();
}
_x1dMenuSize = in.readUint16BE();
_x1d_menu = (int *)malloc(sizeof(int) * _x1dMenuSize);
_y1d_menu = (int *)malloc(sizeof(int) * _x1dMenuSize);
for (i = 0; i < _x1dMenuSize; i++) {
_x1d_menu[i] = in.readSint16BE();
_y1d_menu[i] = in.readSint16BE();
}
_frameXSize = in.readUint16BE();
_frameX = (int *)malloc(sizeof(int) * _frameXSize);
for (i = 0; i < _frameXSize; i++) {
_frameX[i] = in.readSint16BE();
}
_candleXSize = in.readUint16BE();
_candleX = (int *)malloc(sizeof(int) * _candleXSize);
_candleY = (int *)malloc(sizeof(int) * _candleXSize);
for (i = 0; i < _candleXSize; i++) {
_candleX[i] = in.readSint16BE();
_candleY[i] = in.readSint16BE();
}
_pianistXSize = in.readUint16BE();
_pianistX = (int *)malloc(sizeof(int) * _pianistXSize);
for (i = 0; i < _pianistXSize; i++) {
_pianistX[i] = in.readSint16BE();
}
_drunkXSize = in.readUint16BE();
_drunkX = (int *)malloc(sizeof(int) * _drunkXSize);
for (i = 0; i < _drunkXSize; i++) {
_drunkX[i] = in.readSint16BE();
}
_roomPreUpdatesSize = in.readUint16BE();
_roomPreUpdates = (RoomUpdate *)malloc(sizeof(RoomUpdate) * _roomPreUpdatesSize);
for (i = 0; i < _roomPreUpdatesSize; i++) {
_roomPreUpdates[i].roomNum = in.readSint16BE();
_roomPreUpdates[i].flag = in.readSint16BE();
_roomPreUpdates[i].flagValue = in.readSint16BE();
_roomPreUpdates[i].sourceX = in.readSint16BE();
_roomPreUpdates[i].sourceY = in.readSint16BE();
_roomPreUpdates[i].destX = in.readSint16BE();
_roomPreUpdates[i].destY = in.readSint16BE();
_roomPreUpdates[i].width = in.readSint16BE();
_roomPreUpdates[i].height = in.readSint16BE();
_roomPreUpdates[i].type = in.readSint16BE();
}
_roomUpdatesSize = in.readUint16BE();
_roomUpdates = (RoomUpdate *)malloc(sizeof(RoomUpdate) * _roomUpdatesSize);
for (i = 0; i < _roomUpdatesSize; i++) {
_roomUpdates[i].roomNum = in.readSint16BE();
_roomUpdates[i].flag = in.readSint16BE();
_roomUpdates[i].flagValue = in.readSint16BE();
_roomUpdates[i].sourceX = in.readSint16BE();
_roomUpdates[i].sourceY = in.readSint16BE();
_roomUpdates[i].destX = in.readSint16BE();
_roomUpdates[i].destY = in.readSint16BE();
_roomUpdates[i].width = in.readSint16BE();
_roomUpdates[i].height = in.readSint16BE();
_roomUpdates[i].type = in.readSint16BE();
}
_roomActionsSize = in.readUint16BE();
_roomActions = (RoomTalkAction *)malloc(sizeof(RoomTalkAction) * _roomActionsSize);
for (i = 0; i < _roomActionsSize; i++) {
_roomActions[i].room = in.readSint16BE();
_roomActions[i].chapter = in.readSint16BE();
_roomActions[i].action = in.readSint16BE();
_roomActions[i].objectID = in.readSint16BE();
_roomActions[i].speechID = in.readSint16BE();
}
_talkSequencesSize = in.readUint16BE();
_talkSequences = (TalkSequenceCommand *)malloc(sizeof(TalkSequenceCommand) * _talkSequencesSize);
for (i = 0; i < _talkSequencesSize; i++) {
_talkSequences[i].chapter = in.readSint16BE();
_talkSequences[i].sequence = in.readSint16BE();
_talkSequences[i].commandType = in.readSint16BE();
_talkSequences[i].action = in.readSint16BE();
}
_numLangs = in.readUint16BE();
_text = loadTexts(in);
_textd = loadTexts(in);
_textb = loadTexts(in);
_textbj = loadTexts(in);
_texte = loadTexts(in);
_texti = loadTexts(in);
_textl = loadTexts(in);
_textp = loadTexts(in);
_textt = loadTexts(in);
_textvb = loadTexts(in);
_textsys = loadTexts(in);
_texthis = loadTexts(in);
_textverbs = loadTexts(in);
_textmisc = loadTexts(in);
_textd1 = loadTexts(in);
return true;
}
char **DrasculaEngine::loadTexts(Common::File &in) {
int numTexts = in.readUint16BE();
char **res = (char **)malloc(sizeof(char *) * numTexts);
int entryLen;
char *pos = nullptr;
int len;
for (int lang = 0; lang < _numLangs; lang++) {
entryLen = in.readUint16BE();
if (lang == _lang) {
pos = (char *)malloc(entryLen);
in.read(pos, entryLen);
res[0] = pos;
pos += DATAALIGNMENT;
for (int i = 1; i < numTexts; i++) {
pos -= 2;
len = READ_BE_UINT16(pos);
pos += 2 + len;
res[i] = pos;
}
} else
in.seek(entryLen, SEEK_CUR);
}
return res;
}
void DrasculaEngine::freeTexts(char **ptr) {
if (!ptr)
return;
free(*ptr);
free(ptr);
}
} // End of namespace Drascula