2007-05-30 21:56:52 +00:00
|
|
|
/* 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.
|
2004-04-09 12:36:06 +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
|
2004-10-15 06:06:47 +00:00
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2004-04-09 12:36:06 +00:00
|
|
|
* 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
|
2005-10-18 01:30:26 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
2004-04-09 12:36:06 +00:00
|
|
|
*
|
2006-02-09 12:19:53 +00:00
|
|
|
* $URL$
|
|
|
|
* $Id$
|
2004-04-09 12:36:06 +00:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "common/config-manager.h"
|
2005-08-19 22:12:09 +00:00
|
|
|
|
2004-11-11 13:37:35 +00:00
|
|
|
#include "sound/mididrv.h"
|
2007-07-29 16:31:29 +00:00
|
|
|
#include "sound/mixer.h"
|
2006-01-04 07:39:16 +00:00
|
|
|
|
2005-06-24 16:01:42 +00:00
|
|
|
#include "kyra/kyra.h"
|
2007-07-29 16:31:29 +00:00
|
|
|
#include "kyra/sound.h"
|
2005-06-24 16:01:42 +00:00
|
|
|
#include "kyra/resource.h"
|
2005-08-19 22:12:09 +00:00
|
|
|
#include "kyra/screen.h"
|
2006-01-02 22:58:59 +00:00
|
|
|
#include "kyra/text.h"
|
2007-07-29 16:33:11 +00:00
|
|
|
#include "kyra/timer.h"
|
|
|
|
#include "kyra/script.h"
|
2005-08-19 22:12:09 +00:00
|
|
|
|
2006-04-08 11:21:04 +00:00
|
|
|
namespace Kyra {
|
2004-10-15 06:06:47 +00:00
|
|
|
|
2006-11-09 14:38:16 +00:00
|
|
|
KyraEngine::KyraEngine(OSystem *system, const GameFlags &flags)
|
2005-08-19 22:12:09 +00:00
|
|
|
: Engine(system) {
|
2006-04-27 00:39:10 +00:00
|
|
|
_res = 0;
|
|
|
|
_sound = 0;
|
|
|
|
_text = 0;
|
2007-07-29 16:31:29 +00:00
|
|
|
_staticres = 0;
|
2007-07-29 16:33:11 +00:00
|
|
|
_timer = 0;
|
|
|
|
_scriptInterpreter = 0;
|
2007-09-19 08:40:12 +00:00
|
|
|
|
2007-07-29 16:33:11 +00:00
|
|
|
_flags = flags;
|
|
|
|
_gameSpeed = 60;
|
|
|
|
_tickLength = (uint8)(1000.0 / _gameSpeed);
|
2007-09-19 08:40:12 +00:00
|
|
|
|
2006-07-25 15:11:42 +00:00
|
|
|
_quitFlag = false;
|
2007-09-19 08:40:12 +00:00
|
|
|
|
2007-07-29 16:31:29 +00:00
|
|
|
_skipFlag = false;
|
2007-09-19 08:40:12 +00:00
|
|
|
|
2007-10-10 09:06:15 +00:00
|
|
|
_trackMap = 0;
|
|
|
|
_trackMapSize = 0;
|
|
|
|
_lastMusicCommand = -1;
|
|
|
|
_curSfxFile = _curMusicTheme = -1;
|
|
|
|
|
2007-07-29 16:31:29 +00:00
|
|
|
memset(_flagsTable, 0, sizeof(_flagsTable));
|
2007-06-08 23:04:13 +00:00
|
|
|
|
2006-11-13 12:00:36 +00:00
|
|
|
// sets up all engine specific debug levels
|
|
|
|
Common::addSpecialDebugLevel(kDebugLevelScriptFuncs, "ScriptFuncs", "Script function debug level");
|
|
|
|
Common::addSpecialDebugLevel(kDebugLevelScript, "Script", "Script interpreter debug level");
|
|
|
|
Common::addSpecialDebugLevel(kDebugLevelSprites, "Sprites", "Sprite debug level");
|
|
|
|
Common::addSpecialDebugLevel(kDebugLevelScreen, "Screen", "Screen debug level");
|
|
|
|
Common::addSpecialDebugLevel(kDebugLevelSound, "Sound", "Sound debug level");
|
|
|
|
Common::addSpecialDebugLevel(kDebugLevelAnimator, "Animator", "Animator debug level");
|
|
|
|
Common::addSpecialDebugLevel(kDebugLevelMain, "Main", "Generic debug level");
|
|
|
|
Common::addSpecialDebugLevel(kDebugLevelGUI, "GUI", "GUI debug level");
|
|
|
|
Common::addSpecialDebugLevel(kDebugLevelSequence, "Sequence", "Sequence debug level");
|
|
|
|
Common::addSpecialDebugLevel(kDebugLevelMovie, "Movie", "Movie debug level");
|
2007-07-29 16:33:11 +00:00
|
|
|
Common::addSpecialDebugLevel(kDebugLevelTimer, "Timer", "Timer debug level");
|
2007-09-19 13:55:05 +00:00
|
|
|
|
|
|
|
system->getEventManager()->registerRandomSource(_rnd, "kyra");
|
2006-04-27 00:39:10 +00:00
|
|
|
}
|
2004-04-09 12:36:06 +00:00
|
|
|
|
2006-05-03 13:40:21 +00:00
|
|
|
int KyraEngine::init() {
|
2006-04-27 00:39:10 +00:00
|
|
|
// Setup mixer
|
2007-04-15 16:41:20 +00:00
|
|
|
if (!_mixer->isReady())
|
2006-04-27 00:39:10 +00:00
|
|
|
warning("Sound initialization failed.");
|
|
|
|
|
|
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kSFXSoundType, ConfMan.getInt("sfx_volume"));
|
|
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kMusicSoundType, ConfMan.getInt("music_volume"));
|
|
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kSpeechSoundType, ConfMan.getInt("speech_volume"));
|
2007-09-19 08:40:12 +00:00
|
|
|
|
2007-12-26 01:17:58 +00:00
|
|
|
// We prefer AdLib over native MIDI, since our AdLib playback code is much
|
|
|
|
// more mature than our MIDI player. For example we are missing MT-32 support
|
|
|
|
// and it seems our MIDI playback code has threading issues (see bug #1506583
|
|
|
|
// "KYRA1: Crash on exceeded polyphony" for more information).
|
2005-12-30 16:36:53 +00:00
|
|
|
int midiDriver = MidiDriver::detectMusicDriver(MDT_MIDI | MDT_ADLIB/* | MDT_PREFER_MIDI*/);
|
2004-11-14 14:11:54 +00:00
|
|
|
|
2007-12-16 18:48:43 +00:00
|
|
|
if (_flags.platform == Common::kPlatformFMTowns || _flags.platform == Common::kPlatformPC98) {
|
2007-09-15 14:53:21 +00:00
|
|
|
// TODO: currently we don't support the PC98 sound data,
|
|
|
|
// but since it has the FM-Towns data files, we just use the
|
|
|
|
// FM-Towns driver
|
2007-12-21 09:28:27 +00:00
|
|
|
if (_flags.gameID == GI_KYRA1)
|
|
|
|
_sound = new SoundTowns(this, _mixer);
|
|
|
|
else
|
|
|
|
_sound = new SoundTowns_v2(this, _mixer);
|
2007-02-16 16:57:59 +00:00
|
|
|
} else if (midiDriver == MD_ADLIB) {
|
2007-01-26 13:09:48 +00:00
|
|
|
_sound = new SoundAdlibPC(this, _mixer);
|
2006-02-27 22:39:55 +00:00
|
|
|
assert(_sound);
|
|
|
|
} else {
|
|
|
|
bool native_mt32 = ((midiDriver == MD_MT32) || ConfMan.getBool("native_mt32"));
|
|
|
|
|
|
|
|
MidiDriver *driver = MidiDriver::createMidi(midiDriver);
|
|
|
|
assert(driver);
|
2007-04-15 16:41:20 +00:00
|
|
|
if (native_mt32)
|
2006-02-27 22:39:55 +00:00
|
|
|
driver->property(MidiDriver::PROP_CHANNEL_MASK, 0x03FE);
|
2004-11-14 14:11:54 +00:00
|
|
|
|
2007-01-26 13:09:48 +00:00
|
|
|
SoundMidiPC *soundMidiPc = new SoundMidiPC(this, _mixer, driver);
|
2006-02-27 22:39:55 +00:00
|
|
|
_sound = soundMidiPc;
|
|
|
|
assert(_sound);
|
|
|
|
soundMidiPc->hasNativeMT32(native_mt32);
|
2006-08-01 13:42:33 +00:00
|
|
|
|
2006-03-28 15:15:36 +00:00
|
|
|
// Unlike some SCUMM games, it's not that the MIDI sounds are
|
|
|
|
// missing. It's just that at least at the time of writing they
|
2006-08-01 13:42:33 +00:00
|
|
|
// are decidedly inferior to the Adlib ones.
|
2007-12-26 01:17:58 +00:00
|
|
|
if (ConfMan.getBool("multi_midi")) {
|
2007-01-26 13:09:48 +00:00
|
|
|
SoundAdlibPC *adlib = new SoundAdlibPC(this, _mixer);
|
2006-03-28 15:15:36 +00:00
|
|
|
assert(adlib);
|
2007-09-19 08:40:12 +00:00
|
|
|
|
2006-03-28 15:15:36 +00:00
|
|
|
_sound = new MixedSoundDriver(this, _mixer, soundMidiPc, adlib);
|
|
|
|
assert(_sound);
|
|
|
|
}
|
2006-02-27 22:39:55 +00:00
|
|
|
}
|
2007-01-29 18:15:14 +00:00
|
|
|
|
2006-07-08 12:23:44 +00:00
|
|
|
_res = new Resource(this);
|
|
|
|
assert(_res);
|
2007-10-09 05:40:20 +00:00
|
|
|
_res->reset();
|
2006-03-18 14:43:18 +00:00
|
|
|
_staticres = new StaticResource(this);
|
|
|
|
assert(_staticres);
|
2006-05-19 07:34:24 +00:00
|
|
|
if (!_staticres->init())
|
|
|
|
error("_staticres->init() failed");
|
2007-07-29 16:33:11 +00:00
|
|
|
_timer = new TimerManager(this, _system);
|
|
|
|
assert(_timer);
|
|
|
|
_scriptInterpreter = new ScriptHelper(this);
|
|
|
|
assert(_scriptInterpreter);
|
2007-09-19 08:40:12 +00:00
|
|
|
|
2007-07-29 16:33:11 +00:00
|
|
|
setupOpcodeTable();
|
2006-02-09 12:19:53 +00:00
|
|
|
|
2006-08-26 22:17:30 +00:00
|
|
|
_lang = 0;
|
|
|
|
Common::Language lang = Common::parseLanguage(ConfMan.get("language"));
|
|
|
|
|
2007-02-12 18:01:51 +00:00
|
|
|
if (_flags.gameID == GI_KYRA2 || _flags.gameID == GI_KYRA3) {
|
|
|
|
switch (lang) {
|
|
|
|
case Common::EN_ANY:
|
|
|
|
case Common::EN_USA:
|
|
|
|
case Common::EN_GRB:
|
|
|
|
_lang = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Common::FR_FRA:
|
|
|
|
_lang = 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Common::DE_DEU:
|
|
|
|
_lang = 2;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
warning("unsupported language, switching back to English");
|
|
|
|
_lang = 0;
|
|
|
|
break;
|
|
|
|
}
|
2006-08-26 22:17:30 +00:00
|
|
|
}
|
|
|
|
|
2004-11-23 00:03:25 +00:00
|
|
|
return 0;
|
2004-04-09 12:36:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
KyraEngine::~KyraEngine() {
|
2005-08-19 22:12:09 +00:00
|
|
|
delete _res;
|
2006-01-13 23:06:04 +00:00
|
|
|
delete _sound;
|
2006-01-02 22:58:59 +00:00
|
|
|
delete _text;
|
2007-07-29 16:33:11 +00:00
|
|
|
delete _timer;
|
|
|
|
delete _scriptInterpreter;
|
2006-02-09 12:19:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void KyraEngine::quitGame() {
|
2006-03-07 14:16:52 +00:00
|
|
|
debugC(9, kDebugLevelMain, "KyraEngine::quitGame()");
|
2006-05-21 15:41:32 +00:00
|
|
|
_quitFlag = true;
|
|
|
|
// Nothing to do here
|
2006-02-09 12:19:53 +00:00
|
|
|
}
|
|
|
|
|
2007-03-18 18:27:52 +00:00
|
|
|
Common::Point KyraEngine::getMousePos() const {
|
2007-04-01 17:36:13 +00:00
|
|
|
Common::Point mouse = _eventMan->getMousePos();
|
2007-07-29 16:31:29 +00:00
|
|
|
|
2007-03-18 18:27:52 +00:00
|
|
|
if (_flags.useHiResOverlay) {
|
|
|
|
mouse.x >>= 1;
|
|
|
|
mouse.y >>= 1;
|
|
|
|
}
|
2007-04-15 16:41:20 +00:00
|
|
|
|
2007-07-29 16:31:29 +00:00
|
|
|
return mouse;
|
2005-12-08 17:19:18 +00:00
|
|
|
}
|
|
|
|
|
2006-02-11 08:31:13 +00:00
|
|
|
int KyraEngine::setGameFlag(int flag) {
|
|
|
|
_flagsTable[flag >> 3] |= (1 << (flag & 7));
|
|
|
|
return 1;
|
2005-12-10 15:52:38 +00:00
|
|
|
}
|
|
|
|
|
2007-07-29 16:31:29 +00:00
|
|
|
int KyraEngine::queryGameFlag(int flag) const {
|
2006-02-11 08:31:13 +00:00
|
|
|
return ((_flagsTable[flag >> 3] >> (flag & 7)) & 1);
|
2005-12-10 15:52:38 +00:00
|
|
|
}
|
|
|
|
|
2006-02-11 08:31:13 +00:00
|
|
|
int KyraEngine::resetGameFlag(int flag) {
|
|
|
|
_flagsTable[flag >> 3] &= ~(1 << (flag & 7));
|
|
|
|
return 0;
|
2005-12-26 14:53:51 +00:00
|
|
|
}
|
|
|
|
|
2007-07-29 16:31:29 +00:00
|
|
|
void KyraEngine::delayUntil(uint32 timestamp, bool updateTimers, bool update, bool isMainLoop) {
|
|
|
|
while (_system->getMillis() < timestamp && !_quitFlag) {
|
|
|
|
if (timestamp - _system->getMillis() >= 10)
|
|
|
|
delay(10, update, isMainLoop);
|
2005-11-18 23:55:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-29 16:31:29 +00:00
|
|
|
void KyraEngine::delay(uint32 amount, bool update, bool isMainLoop) {
|
|
|
|
_system->delayMillis(amount);
|
2005-12-09 23:02:16 +00:00
|
|
|
}
|
|
|
|
|
2007-07-29 16:31:29 +00:00
|
|
|
void KyraEngine::delayWithTicks(int ticks) {
|
|
|
|
delay(ticks * _tickLength);
|
2005-12-09 23:02:16 +00:00
|
|
|
}
|
2006-04-08 11:21:04 +00:00
|
|
|
|
2005-08-19 22:12:09 +00:00
|
|
|
} // End of namespace Kyra
|
2007-04-15 16:41:20 +00:00
|
|
|
|
2007-12-16 18:48:43 +00:00
|
|
|
|