mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-24 19:45:07 +00:00
769 lines
26 KiB
C++
769 lines
26 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.
|
|
*
|
|
*/
|
|
|
|
#include "sword1/sword1.h"
|
|
|
|
#include "sword1/resman.h"
|
|
#include "sword1/objectman.h"
|
|
#include "sword1/mouse.h"
|
|
#include "sword1/logic.h"
|
|
#include "sword1/sound.h"
|
|
#include "sword1/screen.h"
|
|
#include "sword1/swordres.h"
|
|
#include "sword1/menu.h"
|
|
#include "sword1/music.h"
|
|
#include "sword1/control.h"
|
|
|
|
#include "common/config-manager.h"
|
|
#include "common/textconsole.h"
|
|
|
|
#include "engines/util.h"
|
|
|
|
#include "gui/message.h"
|
|
#include "gui/gui-manager.h"
|
|
|
|
namespace Sword1 {
|
|
|
|
SystemVars SwordEngine::_systemVars;
|
|
|
|
SwordEngine::SwordEngine(OSystem *syst)
|
|
: Engine(syst) {
|
|
|
|
if (!scumm_stricmp(ConfMan.get("gameid").c_str(), "sword1demo") ||
|
|
!scumm_stricmp(ConfMan.get("gameid").c_str(), "sword1psxdemo") ||
|
|
!scumm_stricmp(ConfMan.get("gameid").c_str(), "sword1macdemo"))
|
|
_features = GF_DEMO;
|
|
else
|
|
_features = 0;
|
|
|
|
// Add default file directories
|
|
const Common::FSNode gameDataDir(ConfMan.get("path"));
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "clusters");
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "music");
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "speech");
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "video");
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "smackshi");
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "streams"); // PSX videos
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "english"); // PSX Demo
|
|
SearchMan.addSubDirectoryMatching(gameDataDir, "italian"); // PSX Demo
|
|
|
|
_console = new SwordConsole(this);
|
|
}
|
|
|
|
SwordEngine::~SwordEngine() {
|
|
delete _control;
|
|
delete _logic;
|
|
delete _menu;
|
|
delete _sound;
|
|
delete _music;
|
|
delete _screen;
|
|
delete _mouse;
|
|
delete _objectMan;
|
|
delete _resMan;
|
|
delete _console;
|
|
}
|
|
|
|
Common::Error SwordEngine::init() {
|
|
|
|
initGraphics(640, 480, true);
|
|
|
|
if (0 == scumm_stricmp(ConfMan.get("gameid").c_str(), "sword1mac") ||
|
|
0 == scumm_stricmp(ConfMan.get("gameid").c_str(), "sword1macdemo"))
|
|
_systemVars.platform = Common::kPlatformMacintosh;
|
|
else if (0 == scumm_stricmp(ConfMan.get("gameid").c_str(), "sword1psx") ||
|
|
0 == scumm_stricmp(ConfMan.get("gameid").c_str(), "sword1psxdemo"))
|
|
_systemVars.platform = Common::kPlatformPSX;
|
|
else
|
|
_systemVars.platform = Common::kPlatformWindows;
|
|
|
|
checkCdFiles();
|
|
|
|
debug(5, "Starting resource manager");
|
|
_resMan = new ResMan("swordres.rif", _systemVars.platform == Common::kPlatformMacintosh);
|
|
debug(5, "Starting object manager");
|
|
_objectMan = new ObjectMan(_resMan);
|
|
_mouse = new Mouse(_system, _resMan, _objectMan);
|
|
_screen = new Screen(_system, _resMan, _objectMan);
|
|
_music = new Music(_mixer);
|
|
_sound = new Sound("", _mixer, _resMan);
|
|
_menu = new Menu(_screen, _mouse);
|
|
_logic = new Logic(this, _objectMan, _resMan, _screen, _mouse, _sound, _music, _menu, _system, _mixer);
|
|
_mouse->useLogicAndMenu(_logic, _menu);
|
|
|
|
syncSoundSettings();
|
|
|
|
_systemVars.justRestoredGame = 0;
|
|
_systemVars.currentCD = 0;
|
|
_systemVars.controlPanelMode = CP_NEWGAME;
|
|
_systemVars.forceRestart = false;
|
|
_systemVars.wantFade = true;
|
|
|
|
switch (Common::parseLanguage(ConfMan.get("language"))) {
|
|
case Common::DE_DEU:
|
|
_systemVars.language = BS1_GERMAN;
|
|
break;
|
|
case Common::FR_FRA:
|
|
_systemVars.language = BS1_FRENCH;
|
|
break;
|
|
case Common::IT_ITA:
|
|
_systemVars.language = BS1_ITALIAN;
|
|
break;
|
|
case Common::ES_ESP:
|
|
_systemVars.language = BS1_SPANISH;
|
|
break;
|
|
case Common::PT_BRA:
|
|
_systemVars.language = BS1_PORT;
|
|
break;
|
|
case Common::CZ_CZE:
|
|
_systemVars.language = BS1_CZECH;
|
|
break;
|
|
default:
|
|
_systemVars.language = BS1_ENGLISH;
|
|
}
|
|
|
|
_systemVars.showText = ConfMan.getBool("subtitles");
|
|
|
|
_systemVars.playSpeech = 1;
|
|
_mouseState = 0;
|
|
|
|
// Some Mac versions use big endian for the speech files but not all of them.
|
|
if (_systemVars.platform == Common::kPlatformMacintosh)
|
|
_sound->checkSpeechFileEndianness();
|
|
|
|
_logic->initialize();
|
|
_objectMan->initialize();
|
|
_mouse->initialize();
|
|
_control = new Control(_saveFileMan, _resMan, _objectMan, _system, _mouse, _sound, _music);
|
|
|
|
return Common::kNoError;
|
|
}
|
|
|
|
void SwordEngine::reinitialize() {
|
|
_sound->quitScreen();
|
|
_resMan->flush(); // free everything that's currently alloced and opened. (*evil*)
|
|
|
|
_logic->initialize(); // now reinitialize these objects as they (may) have locked
|
|
_objectMan->initialize(); // resources which have just been wiped.
|
|
_mouse->initialize();
|
|
_system->warpMouse(320, 240);
|
|
_systemVars.wantFade = true;
|
|
}
|
|
|
|
void SwordEngine::syncSoundSettings() {
|
|
Engine::syncSoundSettings();
|
|
|
|
uint musicVol = ConfMan.getInt("music_volume");
|
|
uint sfxVol = ConfMan.getInt("sfx_volume");
|
|
uint speechVol = ConfMan.getInt("speech_volume");
|
|
|
|
uint musicBal = 50;
|
|
if (ConfMan.hasKey("music_balance")) {
|
|
musicBal = CLIP(ConfMan.getInt("music_balance"), 0, 100);
|
|
}
|
|
|
|
uint speechBal = 50;
|
|
if (ConfMan.hasKey("speech_balance")) {
|
|
speechBal = CLIP(ConfMan.getInt("speech_balance"), 0, 100);
|
|
}
|
|
uint sfxBal = 50;
|
|
if (ConfMan.hasKey("sfx_balance")) {
|
|
sfxBal = CLIP(ConfMan.getInt("sfx_balance"), 0, 100);
|
|
}
|
|
|
|
uint musicVolL = 2 * musicVol * musicBal / 100;
|
|
uint musicVolR = 2 * musicVol - musicVolL;
|
|
|
|
uint speechVolL = 2 * speechVol * speechBal / 100;
|
|
uint speechVolR = 2 * speechVol - speechVolL;
|
|
|
|
uint sfxVolL = 2 * sfxVol * sfxBal / 100;
|
|
uint sfxVolR = 2 * sfxVol - sfxVolL;
|
|
|
|
if (musicVolR > 255) {
|
|
musicVolR = 255;
|
|
}
|
|
if (musicVolL > 255) {
|
|
musicVolL = 255;
|
|
}
|
|
|
|
if (speechVolR > 255) {
|
|
speechVolR = 255;
|
|
}
|
|
if (speechVolL > 255) {
|
|
speechVolL = 255;
|
|
}
|
|
if (sfxVolR > 255) {
|
|
sfxVolR = 255;
|
|
}
|
|
if (sfxVolL > 255) {
|
|
sfxVolL = 255;
|
|
}
|
|
|
|
bool mute = ConfMan.getBool("mute");
|
|
|
|
if (mute) {
|
|
_music->setVolume(0, 0);
|
|
_sound->setSpeechVol(0, 0);
|
|
_sound->setSfxVol(0, 0);
|
|
} else {
|
|
_music->setVolume(musicVolL, musicVolR);
|
|
_sound->setSpeechVol(speechVolL, speechVolR);
|
|
_sound->setSfxVol(sfxVolL, sfxVolR);
|
|
}
|
|
}
|
|
|
|
void SwordEngine::flagsToBool(bool *dest, uint8 flags) {
|
|
uint8 bitPos = 0;
|
|
while (flags) {
|
|
if (flags & 1)
|
|
dest[bitPos] = true;
|
|
flags >>= 1;
|
|
bitPos++;
|
|
}
|
|
}
|
|
|
|
static const char *const errorMsgs[] = {
|
|
"The file \"%s\" is missing and the game doesn't work without it.\n"
|
|
"Please copy it from CD %d and try starting the game again.\n"
|
|
"The Readme file also contains further information.",
|
|
|
|
"%d important files are missing, the game can't start without them.\n"
|
|
"Please copy these files from their corresponding CDs:\n",
|
|
|
|
"The file \"%s\" is missing.\n"
|
|
"Even though the game may initially seem to\n"
|
|
"work fine, it will crash when it needs the\n"
|
|
"data from this file and you will be thrown back to your last savegame.\n"
|
|
"Please copy the file from CD %d and start the game again.",
|
|
|
|
"%d files are missing.\n"
|
|
"Even though the game may initially seem to\n"
|
|
"work fine, it will crash when it needs the\n"
|
|
"data from these files and you will be thrown back to your last savegame.\n"
|
|
"Please copy these files from their corresponding CDs:\n"
|
|
};
|
|
|
|
const CdFile SwordEngine::_pcCdFileList[] = {
|
|
{ "paris2.clu", FLAG_CD1 },
|
|
{ "ireland.clu", FLAG_CD2 },
|
|
{ "paris3.clu", FLAG_CD1 },
|
|
{ "paris4.clu", FLAG_CD1 },
|
|
{ "scotland.clu", FLAG_CD2 },
|
|
{ "spain.clu", FLAG_CD2 },
|
|
{ "syria.clu", FLAG_CD2 },
|
|
{ "train.clu", FLAG_CD2 },
|
|
{ "compacts.clu", FLAG_CD1 | FLAG_DEMO | FLAG_IMMED },
|
|
{ "general.clu", FLAG_CD1 | FLAG_DEMO | FLAG_IMMED },
|
|
{ "maps.clu", FLAG_CD1 | FLAG_DEMO },
|
|
{ "paris1.clu", FLAG_CD1 | FLAG_DEMO },
|
|
{ "scripts.clu", FLAG_CD1 | FLAG_DEMO | FLAG_IMMED },
|
|
{ "swordres.rif", FLAG_CD1 | FLAG_DEMO | FLAG_IMMED },
|
|
{ "text.clu", FLAG_CD1 | FLAG_DEMO },
|
|
{ "cows.mad", FLAG_DEMO },
|
|
{ "speech1.clu", FLAG_SPEECH1 },
|
|
{ "speech2.clu", FLAG_SPEECH2 }
|
|
#ifdef USE_FLAC
|
|
, { "speech1.clf", FLAG_SPEECH1 },
|
|
{ "speech2.clf", FLAG_SPEECH2 }
|
|
#endif
|
|
#ifdef USE_VORBIS
|
|
, { "speech1.clv", FLAG_SPEECH1 },
|
|
{ "speech2.clv", FLAG_SPEECH2 }
|
|
#endif
|
|
#ifdef USE_MAD
|
|
, { "speech1.cl3", FLAG_SPEECH1 },
|
|
{ "speech2.cl3", FLAG_SPEECH2 }
|
|
#endif
|
|
};
|
|
|
|
const CdFile SwordEngine::_macCdFileList[] = {
|
|
{ "paris2.clm", FLAG_CD1 },
|
|
{ "ireland.clm", FLAG_CD2 },
|
|
{ "paris3.clm", FLAG_CD1 },
|
|
{ "paris4.clm", FLAG_CD1 },
|
|
{ "scotland.clm", FLAG_CD2 },
|
|
{ "spain.clm", FLAG_CD2 },
|
|
{ "syria.clm", FLAG_CD2 },
|
|
{ "train.clm", FLAG_CD2 },
|
|
{ "compacts.clm", FLAG_CD1 | FLAG_DEMO | FLAG_IMMED },
|
|
{ "general.clm", FLAG_CD1 | FLAG_DEMO | FLAG_IMMED },
|
|
{ "maps.clm", FLAG_CD1 | FLAG_DEMO },
|
|
{ "paris1.clm", FLAG_CD1 | FLAG_DEMO },
|
|
{ "scripts.clm", FLAG_CD1 | FLAG_DEMO | FLAG_IMMED },
|
|
{ "swordres.rif", FLAG_CD1 | FLAG_DEMO | FLAG_IMMED },
|
|
{ "text.clm", FLAG_CD1 | FLAG_DEMO },
|
|
{ "speech1.clu", FLAG_SPEECH1 },
|
|
{ "speech2.clu", FLAG_SPEECH2 }
|
|
#ifdef USE_FLAC
|
|
,{ "speech1.clf", FLAG_SPEECH1 },
|
|
{ "speech2.clf", FLAG_SPEECH2 }
|
|
#endif
|
|
#ifdef USE_VORBIS
|
|
,{ "speech1.clv", FLAG_SPEECH1 },
|
|
{ "speech2.clv", FLAG_SPEECH2 }
|
|
#endif
|
|
#ifdef USE_MAD
|
|
,{ "speech1.cl3", FLAG_SPEECH1 },
|
|
{ "speech2.cl3", FLAG_SPEECH2 }
|
|
#endif
|
|
};
|
|
|
|
const CdFile SwordEngine::_psxCdFileList[] = { // PSX edition has only one cd
|
|
{ "paris2.clu", FLAG_CD1 },
|
|
{ "ireland.clu", FLAG_CD1 },
|
|
{ "paris3.clu", FLAG_CD1 },
|
|
{ "paris4.clu", FLAG_CD1 },
|
|
{ "scotland.clu", FLAG_CD1 },
|
|
{ "spain.clu", FLAG_CD1 },
|
|
{ "syria.clu", FLAG_CD1 },
|
|
{ "train.clu", FLAG_CD1 },
|
|
{ "train.plx", FLAG_CD1 },
|
|
{ "compacts.clu", FLAG_CD1 | FLAG_DEMO | FLAG_IMMED },
|
|
{ "general.clu", FLAG_CD1 | FLAG_DEMO | FLAG_IMMED },
|
|
{ "maps.clu", FLAG_CD1 | FLAG_DEMO },
|
|
{ "paris1.clu", FLAG_CD1 | FLAG_DEMO},
|
|
{ "scripts.clu", FLAG_CD1 | FLAG_DEMO | FLAG_IMMED },
|
|
{ "swordres.rif", FLAG_CD1 | FLAG_DEMO | FLAG_IMMED },
|
|
{ "text.clu", FLAG_CD1 | FLAG_DEMO },
|
|
{ "speech.dat", FLAG_SPEECH1 | FLAG_DEMO },
|
|
{ "speech.tab", FLAG_SPEECH1 | FLAG_DEMO },
|
|
{ "speech.inf", FLAG_SPEECH1 | FLAG_DEMO },
|
|
{ "speech.lis", FLAG_SPEECH1 | FLAG_DEMO }
|
|
};
|
|
|
|
void SwordEngine::showFileErrorMsg(uint8 type, bool *fileExists) {
|
|
char msg[1024];
|
|
int missCnt = 0, missNum = 0;
|
|
|
|
if (SwordEngine::isMac()) {
|
|
for (int i = 0; i < ARRAYSIZE(_macCdFileList); i++)
|
|
if (!fileExists[i]) {
|
|
missCnt++;
|
|
missNum = i;
|
|
}
|
|
assert(missCnt > 0); // this function shouldn't get called if there's nothing missing.
|
|
warning("%d files missing", missCnt);
|
|
int msgId = (type == TYPE_IMMED) ? 0 : 2;
|
|
if (missCnt == 1) {
|
|
sprintf(msg, errorMsgs[msgId],
|
|
_macCdFileList[missNum].name, (_macCdFileList[missNum].flags & FLAG_CD2) ? 2 : 1);
|
|
warning("%s", msg);
|
|
} else {
|
|
char *pos = msg + sprintf(msg, errorMsgs[msgId + 1], missCnt);
|
|
warning("%s", msg);
|
|
for (int i = 0; i < ARRAYSIZE(_macCdFileList); i++)
|
|
if (!fileExists[i]) {
|
|
warning("\"%s\" (CD %d)", _macCdFileList[i].name, (_macCdFileList[i].flags & FLAG_CD2) ? 2 : 1);
|
|
pos += sprintf(pos, "\"%s\" (CD %d)\n", _macCdFileList[i].name, (_macCdFileList[i].flags & FLAG_CD2) ? 2 : 1);
|
|
}
|
|
}
|
|
} else if (SwordEngine::isPsx()) {
|
|
for (int i = 0; i < ARRAYSIZE(_psxCdFileList); i++)
|
|
if (!fileExists[i]) {
|
|
missCnt++;
|
|
missNum = i;
|
|
}
|
|
assert(missCnt > 0); // this function shouldn't get called if there's nothing missing.
|
|
warning("%d files missing", missCnt);
|
|
int msgId = (type == TYPE_IMMED) ? 0 : 2;
|
|
if (missCnt == 1) {
|
|
sprintf(msg, errorMsgs[msgId], _psxCdFileList[missNum].name, 1);
|
|
warning("%s", msg);
|
|
} else {
|
|
char *pos = msg + sprintf(msg, errorMsgs[msgId + 1], missCnt);
|
|
warning("%s", msg);
|
|
for (int i = 0; i < ARRAYSIZE(_psxCdFileList); i++)
|
|
if (!fileExists[i]) {
|
|
warning("\"%s\"", _macCdFileList[i].name);
|
|
pos += sprintf(pos, "\"%s\"\n", _macCdFileList[i].name);
|
|
}
|
|
}
|
|
} else {
|
|
for (int i = 0; i < ARRAYSIZE(_pcCdFileList); i++)
|
|
if (!fileExists[i]) {
|
|
missCnt++;
|
|
missNum = i;
|
|
}
|
|
assert(missCnt > 0); // this function shouldn't get called if there's nothing missing.
|
|
warning("%d files missing", missCnt);
|
|
int msgId = (type == TYPE_IMMED) ? 0 : 2;
|
|
if (missCnt == 1) {
|
|
sprintf(msg, errorMsgs[msgId],
|
|
_pcCdFileList[missNum].name, (_pcCdFileList[missNum].flags & FLAG_CD2) ? 2 : 1);
|
|
warning("%s", msg);
|
|
} else {
|
|
char *pos = msg + sprintf(msg, errorMsgs[msgId + 1], missCnt);
|
|
warning("%s", msg);
|
|
for (int i = 0; i < ARRAYSIZE(_pcCdFileList); i++)
|
|
if (!fileExists[i]) {
|
|
warning("\"%s\" (CD %d)", _pcCdFileList[i].name, (_pcCdFileList[i].flags & FLAG_CD2) ? 2 : 1);
|
|
pos += sprintf(pos, "\"%s\" (CD %d)\n", _pcCdFileList[i].name, (_pcCdFileList[i].flags & FLAG_CD2) ? 2 : 1);
|
|
}
|
|
}
|
|
}
|
|
GUI::MessageDialog dialog(msg);
|
|
dialog.runModal();
|
|
if (type == TYPE_IMMED) // we can't start without this file, so error() out.
|
|
error("%s", msg);
|
|
}
|
|
|
|
void SwordEngine::checkCdFiles() { // check if we're running from cd, hdd or what...
|
|
bool fileExists[30];
|
|
bool isFullVersion = false; // default to demo version
|
|
bool missingTypes[8] = { false, false, false, false, false, false, false, false };
|
|
bool foundTypes[8] = { false, false, false, false, false, false, false, false };
|
|
bool cd2FilesFound = false;
|
|
_systemVars.runningFromCd = false;
|
|
_systemVars.playSpeech = true;
|
|
|
|
// check all files and look out if we can find a file that wouldn't exist if this was the demo version
|
|
if (SwordEngine::isMac()) {
|
|
for (int fcnt = 0; fcnt < ARRAYSIZE(_macCdFileList); fcnt++) {
|
|
if (Common::File::exists(_macCdFileList[fcnt].name)) {
|
|
fileExists[fcnt] = true;
|
|
flagsToBool(foundTypes, _macCdFileList[fcnt].flags);
|
|
if (!(_macCdFileList[fcnt].flags & FLAG_DEMO))
|
|
isFullVersion = true;
|
|
if (_macCdFileList[fcnt].flags & FLAG_CD2)
|
|
cd2FilesFound = true;
|
|
} else {
|
|
flagsToBool(missingTypes, _macCdFileList[fcnt].flags);
|
|
fileExists[fcnt] = false;
|
|
}
|
|
}
|
|
} else if (SwordEngine::isPsx()) {
|
|
for (int fcnt = 0; fcnt < ARRAYSIZE(_psxCdFileList); fcnt++) {
|
|
if (Common::File::exists(_psxCdFileList[fcnt].name)) {
|
|
fileExists[fcnt] = true;
|
|
flagsToBool(foundTypes, _psxCdFileList[fcnt].flags);
|
|
if (!(_psxCdFileList[fcnt].flags & FLAG_DEMO))
|
|
isFullVersion = true;
|
|
cd2FilesFound = true;
|
|
} else {
|
|
flagsToBool(missingTypes, _psxCdFileList[fcnt].flags);
|
|
fileExists[fcnt] = false;
|
|
}
|
|
}
|
|
} else {
|
|
for (int fcnt = 0; fcnt < ARRAYSIZE(_pcCdFileList); fcnt++) {
|
|
if (Common::File::exists(_pcCdFileList[fcnt].name)) {
|
|
fileExists[fcnt] = true;
|
|
flagsToBool(foundTypes, _pcCdFileList[fcnt].flags);
|
|
if (!(_pcCdFileList[fcnt].flags & FLAG_DEMO))
|
|
isFullVersion = true;
|
|
if (_pcCdFileList[fcnt].flags & FLAG_CD2)
|
|
cd2FilesFound = true;
|
|
} else {
|
|
flagsToBool(missingTypes, _pcCdFileList[fcnt].flags);
|
|
fileExists[fcnt] = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (((_features & GF_DEMO) == 0) != isFullVersion) // shouldn't happen...
|
|
warning("Your Broken Sword 1 version looks like a %s version but you are starting it as a %s version", isFullVersion ? "full" : "demo", (_features & GF_DEMO) ? "demo" : "full");
|
|
|
|
if (foundTypes[TYPE_SPEECH1]) // we found some kind of speech1 file (.clu, .cl3, .clv)
|
|
missingTypes[TYPE_SPEECH1] = false; // so we don't care if there's a different kind missing
|
|
if (foundTypes[TYPE_SPEECH2]) // same for speech2
|
|
missingTypes[TYPE_SPEECH2] = false;
|
|
|
|
if (isFullVersion) // if this is the full version...
|
|
missingTypes[TYPE_DEMO] = false; // then we don't need demo files...
|
|
else // and vice versa
|
|
missingTypes[TYPE_SPEECH1] = missingTypes[TYPE_SPEECH2] = missingTypes[TYPE_CD1] = missingTypes[TYPE_CD2] = false;
|
|
|
|
bool somethingMissing = false;
|
|
for (int i = 0; i < 8; i++)
|
|
somethingMissing |= missingTypes[i];
|
|
if (somethingMissing) { // okay, there *are* files missing
|
|
// first, update the fileExists[] array depending on our changed missingTypes
|
|
if (SwordEngine::isMac()) {
|
|
for (int fileCnt = 0; fileCnt < ARRAYSIZE(_macCdFileList); fileCnt++)
|
|
if (!fileExists[fileCnt]) {
|
|
fileExists[fileCnt] = true;
|
|
for (int flagCnt = 0; flagCnt < 8; flagCnt++)
|
|
if (missingTypes[flagCnt] && ((_macCdFileList[fileCnt].flags & (1 << flagCnt)) != 0))
|
|
fileExists[fileCnt] = false; // this is one of the files we were looking for
|
|
}
|
|
} else if (SwordEngine::isPsx()) {
|
|
for (int fileCnt = 0; fileCnt < ARRAYSIZE(_psxCdFileList); fileCnt++)
|
|
if (!fileExists[fileCnt]) {
|
|
fileExists[fileCnt] = true;
|
|
for (int flagCnt = 0; flagCnt < 8; flagCnt++)
|
|
if (missingTypes[flagCnt] && ((_psxCdFileList[fileCnt].flags & (1 << flagCnt)) != 0))
|
|
fileExists[fileCnt] = false; // this is one of the files we were looking for
|
|
}
|
|
} else {
|
|
for (int fileCnt = 0; fileCnt < ARRAYSIZE(_pcCdFileList); fileCnt++)
|
|
if (!fileExists[fileCnt]) {
|
|
fileExists[fileCnt] = true;
|
|
for (int flagCnt = 0; flagCnt < 8; flagCnt++)
|
|
if (missingTypes[flagCnt] && ((_pcCdFileList[fileCnt].flags & (1 << flagCnt)) != 0))
|
|
fileExists[fileCnt] = false; // this is one of the files we were looking for
|
|
}
|
|
}
|
|
if (missingTypes[TYPE_IMMED]) {
|
|
// important files missing, can't start the game without them
|
|
showFileErrorMsg(TYPE_IMMED, fileExists);
|
|
} else if ((!missingTypes[TYPE_CD1]) && !cd2FilesFound) {
|
|
/* we have all the data from cd one, but not a single one from CD2.
|
|
I'm not sure how we should handle this, for now I'll just assume that the
|
|
user has set up the extrapath correctly and copied the necessary files to HDD.
|
|
A quite optimistic assumption, I'd say. Maybe we should change this for the release
|
|
to warn the user? */
|
|
warning("CD2 data files not found. I hope you know what you're doing and that\n"
|
|
"you have set up the extrapath and additional data correctly.\n"
|
|
"If you didn't, you should better read the ScummVM readme file");
|
|
_systemVars.runningFromCd = true;
|
|
_systemVars.playSpeech = true;
|
|
} else if (missingTypes[TYPE_CD1] || missingTypes[TYPE_CD2]) {
|
|
// several files from CD1 both CDs are missing. we can probably start, but it'll crash sooner or later
|
|
showFileErrorMsg(TYPE_CD1, fileExists);
|
|
} else if (missingTypes[TYPE_SPEECH1] || missingTypes[TYPE_SPEECH2]) {
|
|
// not so important, but there won't be any voices
|
|
if (missingTypes[TYPE_SPEECH1] && missingTypes[TYPE_SPEECH2])
|
|
warning("Unable to find the speech files. The game will work, but you won't hear any voice output.\n"
|
|
"Please copy the SPEECH.CLU files from both CDs and rename them to SPEECH1.CLU and SPEECH2.CLU,\n"
|
|
"corresponding to the CD number.\n"
|
|
"Please read the ScummVM Readme file for more information");
|
|
else
|
|
warning("Unable to find the speech file from CD %d.\n"
|
|
"You won't hear any voice output in that part of the game.\n"
|
|
"Please read the ScummVM Readme file for more information", missingTypes[TYPE_SPEECH1] ? 1 : 2);
|
|
} else if (missingTypes[TYPE_DEMO]) {
|
|
// for the demo version, we simply expect to have all files immediately
|
|
showFileErrorMsg(TYPE_IMMED, fileExists);
|
|
}
|
|
} // everything's fine, let's play.
|
|
/*if (!isFullVersion)
|
|
_systemVars.isDemo = true;
|
|
*/
|
|
// make the demo flag depend on the Gamesettings for now, and not on what the datafiles look like
|
|
_systemVars.isDemo = (_features & GF_DEMO) != 0;
|
|
}
|
|
|
|
Common::Error SwordEngine::go() {
|
|
_control->checkForOldSaveGames();
|
|
setTotalPlayTime(0);
|
|
|
|
uint16 startPos = ConfMan.getInt("boot_param");
|
|
_control->readSavegameDescriptions();
|
|
if (startPos) {
|
|
_logic->startPositions(startPos);
|
|
} else {
|
|
int saveSlot = ConfMan.getInt("save_slot");
|
|
// Savegames are numbered starting from 1 in the dialog window,
|
|
// but their filenames are numbered starting from 0.
|
|
if (saveSlot >= 0 && _control->savegamesExist() && _control->restoreGameFromFile(saveSlot)) {
|
|
_control->doRestore();
|
|
} else if (_control->savegamesExist()) {
|
|
_systemVars.controlPanelMode = CP_NEWGAME;
|
|
if (_control->runPanel() == CONTROL_GAME_RESTORED)
|
|
_control->doRestore();
|
|
else if (!shouldQuit())
|
|
_logic->startPositions(0);
|
|
} else {
|
|
// no savegames, start new game.
|
|
_logic->startPositions(0);
|
|
}
|
|
}
|
|
_systemVars.controlPanelMode = CP_NORMAL;
|
|
|
|
while (!shouldQuit()) {
|
|
uint8 action = mainLoop();
|
|
|
|
if (!shouldQuit()) {
|
|
// the mainloop was left, we have to reinitialize.
|
|
reinitialize();
|
|
if (action == CONTROL_GAME_RESTORED)
|
|
_control->doRestore();
|
|
else if (action == CONTROL_RESTART_GAME)
|
|
_logic->startPositions(1);
|
|
_systemVars.forceRestart = false;
|
|
_systemVars.controlPanelMode = CP_NORMAL;
|
|
}
|
|
}
|
|
|
|
return Common::kNoError;
|
|
}
|
|
|
|
void SwordEngine::checkCd() {
|
|
uint8 needCd = _cdList[Logic::_scriptVars[NEW_SCREEN]];
|
|
if (_systemVars.runningFromCd) { // are we running from cd?
|
|
if (needCd == 0) { // needCd == 0 means we can use either CD1 or CD2.
|
|
if (_systemVars.currentCD == 0) {
|
|
_systemVars.currentCD = 1; // if there is no CD currently inserted, ask for CD1.
|
|
_control->askForCd();
|
|
} // else: there is already a cd inserted and we don't care if it's cd1 or cd2.
|
|
} else if (needCd != _systemVars.currentCD) { // we need a different CD than the one in drive.
|
|
_music->startMusic(0, 0); //
|
|
_sound->closeCowSystem(); // close music and sound files before changing CDs
|
|
_systemVars.currentCD = needCd; // askForCd will ask the player to insert _systemVars.currentCd,
|
|
_control->askForCd(); // so it has to be updated before calling it.
|
|
}
|
|
} else { // we're running from HDD, we don't have to care about music files and Sound will take care of
|
|
if (needCd) // switching sound.clu files on Sound::newScreen by itself, so there's nothing to be done.
|
|
_systemVars.currentCD = needCd;
|
|
else if (_systemVars.currentCD == 0)
|
|
_systemVars.currentCD = 1;
|
|
}
|
|
}
|
|
|
|
uint8 SwordEngine::mainLoop() {
|
|
uint8 retCode = 0;
|
|
_keyPressed.reset();
|
|
|
|
while ((retCode == 0) && (!shouldQuit())) {
|
|
// do we need the section45-hack from sword.c here?
|
|
checkCd();
|
|
|
|
_screen->newScreen(Logic::_scriptVars[NEW_SCREEN]);
|
|
_logic->newScreen(Logic::_scriptVars[NEW_SCREEN]);
|
|
_sound->newScreen(Logic::_scriptVars[NEW_SCREEN]);
|
|
Logic::_scriptVars[SCREEN] = Logic::_scriptVars[NEW_SCREEN];
|
|
|
|
do {
|
|
uint32 newTime;
|
|
bool scrollFrameShown = false;
|
|
|
|
uint32 frameTime = _system->getMillis();
|
|
_logic->engine();
|
|
_logic->updateScreenParams(); // sets scrolling
|
|
|
|
_screen->draw();
|
|
_mouse->animate();
|
|
_sound->engine();
|
|
_menu->refresh(MENU_TOP);
|
|
_menu->refresh(MENU_BOT);
|
|
|
|
newTime = _system->getMillis();
|
|
if (newTime - frameTime < 1000 / FRAME_RATE) {
|
|
scrollFrameShown = _screen->showScrollFrame();
|
|
delay((1000 / (FRAME_RATE * 2)) - (_system->getMillis() - frameTime));
|
|
}
|
|
|
|
newTime = _system->getMillis();
|
|
if ((newTime - frameTime < 1000 / FRAME_RATE) || (!scrollFrameShown))
|
|
_screen->updateScreen();
|
|
delay((1000 / FRAME_RATE) - (_system->getMillis() - frameTime));
|
|
|
|
_mouse->engine(_mouseCoord.x, _mouseCoord.y, _mouseState);
|
|
|
|
if (_systemVars.forceRestart)
|
|
retCode = CONTROL_RESTART_GAME;
|
|
|
|
// The control panel is triggered by F5 or ESC.
|
|
else if (((_keyPressed.keycode == Common::KEYCODE_F5 || _keyPressed.keycode == Common::KEYCODE_ESCAPE)
|
|
&& (Logic::_scriptVars[MOUSE_STATUS] & 1)) || (_systemVars.controlPanelMode)) {
|
|
retCode = _control->runPanel();
|
|
if (retCode == CONTROL_NOTHING_DONE)
|
|
_screen->fullRefresh();
|
|
}
|
|
|
|
// Check for Debugger Activation
|
|
if (_keyPressed.hasFlags(Common::KBD_CTRL) && _keyPressed.keycode == Common::KEYCODE_d) {
|
|
this->getDebugger()->attach();
|
|
this->getDebugger()->onFrame();
|
|
}
|
|
|
|
_mouseState = 0;
|
|
_keyPressed.reset();
|
|
} while ((Logic::_scriptVars[SCREEN] == Logic::_scriptVars[NEW_SCREEN]) && (retCode == 0) && (!shouldQuit()));
|
|
|
|
if ((retCode == 0) && (Logic::_scriptVars[SCREEN] != 53) && _systemVars.wantFade && (!shouldQuit())) {
|
|
_screen->fadeDownPalette();
|
|
int32 relDelay = (int32)_system->getMillis();
|
|
while (_screen->stillFading()) {
|
|
relDelay += (1000 / FRAME_RATE);
|
|
_screen->updateScreen();
|
|
delay(relDelay - (int32)_system->getMillis());
|
|
}
|
|
}
|
|
|
|
_sound->quitScreen();
|
|
_screen->quitScreen(); // close graphic resources
|
|
_objectMan->closeSection(Logic::_scriptVars[SCREEN]); // close the section that PLAYER has just left, if it's empty now
|
|
}
|
|
return retCode;
|
|
}
|
|
|
|
void SwordEngine::delay(int32 amount) { //copied and mutilated from sky.cpp
|
|
|
|
Common::Event event;
|
|
uint32 start = _system->getMillis();
|
|
|
|
do {
|
|
while (_eventMan->pollEvent(event)) {
|
|
switch (event.type) {
|
|
case Common::EVENT_KEYDOWN:
|
|
_keyPressed = event.kbd;
|
|
break;
|
|
case Common::EVENT_MOUSEMOVE:
|
|
_mouseCoord = event.mouse;
|
|
break;
|
|
case Common::EVENT_LBUTTONDOWN:
|
|
_mouseState |= BS1L_BUTTON_DOWN;
|
|
_mouseCoord = event.mouse;
|
|
break;
|
|
case Common::EVENT_RBUTTONDOWN:
|
|
_mouseState |= BS1R_BUTTON_DOWN;
|
|
_mouseCoord = event.mouse;
|
|
break;
|
|
case Common::EVENT_LBUTTONUP:
|
|
_mouseState |= BS1L_BUTTON_UP;
|
|
_mouseCoord = event.mouse;
|
|
break;
|
|
case Common::EVENT_RBUTTONUP:
|
|
_mouseState |= BS1R_BUTTON_UP;
|
|
_mouseCoord = event.mouse;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
_system->updateScreen();
|
|
|
|
if (amount > 0)
|
|
_system->delayMillis(10);
|
|
|
|
} while (_system->getMillis() < start + amount);
|
|
}
|
|
|
|
bool SwordEngine::mouseIsActive() {
|
|
return Logic::_scriptVars[MOUSE_STATUS] & 1;
|
|
}
|
|
|
|
// The following function is needed to restore proper status after GMM load game
|
|
void SwordEngine::reinitRes() {
|
|
checkCd(); // Reset currentCD var to correct value
|
|
_screen->newScreen(Logic::_scriptVars[NEW_SCREEN]);
|
|
_logic->newScreen(Logic::_scriptVars[NEW_SCREEN]);
|
|
_sound->newScreen(Logic::_scriptVars[NEW_SCREEN]);
|
|
Logic::_scriptVars[SCREEN] = Logic::_scriptVars[NEW_SCREEN];
|
|
_screen->fullRefresh();
|
|
_screen->draw();
|
|
}
|
|
|
|
} // End of namespace Sword1
|