2014-04-18 22:08:34 +02: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.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2018-06-17 00:07:36 +10:00
|
|
|
#include "common/config-manager.h"
|
2014-04-18 22:08:34 +02:00
|
|
|
#include "illusions/illusions.h"
|
|
|
|
#include "illusions/sound.h"
|
2018-05-01 20:33:10 +10:00
|
|
|
#include "audio/midiparser.h"
|
2014-04-18 22:08:34 +02:00
|
|
|
|
|
|
|
namespace Illusions {
|
|
|
|
|
|
|
|
// MusicPlayer
|
|
|
|
|
2014-04-26 00:08:37 +02:00
|
|
|
MusicPlayer::MusicPlayer()
|
|
|
|
: _musicId(0), _flags(0) {
|
2014-04-18 22:08:34 +02:00
|
|
|
_flags = 1; // TODO?
|
|
|
|
}
|
|
|
|
|
|
|
|
MusicPlayer::~MusicPlayer() {
|
2014-04-18 23:12:10 +02:00
|
|
|
stop();
|
2014-04-18 22:08:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void MusicPlayer::play(uint32 musicId, bool looping, int16 volume, int16 pan) {
|
2015-11-24 00:10:10 +01:00
|
|
|
debug(1, "MusicPlayer::play(%08X)", musicId);
|
2014-04-18 22:08:34 +02:00
|
|
|
if (_flags & 1) {
|
|
|
|
stop();
|
|
|
|
_musicId = musicId;
|
|
|
|
_flags |= 2;
|
|
|
|
_flags &= ~4;
|
|
|
|
if (looping) {
|
|
|
|
_flags |= 8;
|
|
|
|
} else {
|
|
|
|
_flags &= ~8;
|
|
|
|
}
|
2014-04-18 23:12:10 +02:00
|
|
|
Common::String filename = Common::String::format("%08x.wav", _musicId);
|
2014-04-18 22:08:34 +02:00
|
|
|
Common::File *fd = new Common::File();
|
|
|
|
fd->open(filename);
|
|
|
|
Audio::AudioStream *audioStream = Audio::makeLoopingAudioStream(Audio::makeWAVStream(fd, DisposeAfterUse::YES), looping ? 0 : 1);
|
2014-04-26 00:08:37 +02:00
|
|
|
g_system->getMixer()->playStream(Audio::Mixer::kMusicSoundType, &_soundHandle, audioStream, -1, volume, pan);
|
2014-04-18 22:08:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void MusicPlayer::stop() {
|
2015-11-24 00:10:10 +01:00
|
|
|
debug(1, "MusicPlayer::stop()");
|
2014-04-18 22:08:34 +02:00
|
|
|
if ((_flags & 1) && (_flags & 2)) {
|
2014-04-26 00:08:37 +02:00
|
|
|
if (g_system->getMixer()->isSoundHandleActive(_soundHandle))
|
|
|
|
g_system->getMixer()->stopHandle(_soundHandle);
|
2014-04-18 22:08:34 +02:00
|
|
|
_flags &= ~2;
|
|
|
|
_flags &= ~4;
|
|
|
|
_flags &= ~8;
|
|
|
|
_musicId = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MusicPlayer::isPlaying() {
|
2014-04-26 00:08:37 +02:00
|
|
|
return (_flags & 1) && (_flags & 2) && g_system->getMixer()->isSoundHandleActive(_soundHandle);
|
2014-04-18 22:08:34 +02:00
|
|
|
}
|
|
|
|
|
2018-05-01 20:33:10 +10:00
|
|
|
// MidiPlayer
|
|
|
|
|
|
|
|
MidiPlayer::MidiPlayer() {
|
|
|
|
MidiDriver::DeviceHandle dev = MidiDriver::detectDevice(MDT_MIDI | MDT_ADLIB | MDT_PREFER_GM);
|
|
|
|
_driver = MidiDriver::createMidi(dev);
|
|
|
|
assert(_driver);
|
|
|
|
_paused = false;
|
|
|
|
|
|
|
|
|
|
|
|
int ret = _driver->open();
|
|
|
|
if (ret == 0) {
|
|
|
|
_driver->sendGMReset();
|
|
|
|
|
|
|
|
_driver->setTimerCallback(this, &timerCallback);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void MidiPlayer::play(const Common::String &filename) {
|
|
|
|
Common::StackLock lock(_mutex);
|
|
|
|
|
|
|
|
stop();
|
|
|
|
|
|
|
|
Common::File *fd = new Common::File();
|
|
|
|
if (!fd->open(filename)) {
|
|
|
|
delete fd;
|
|
|
|
error("MidiPlayer::play() Could not load %s", filename.c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32 size = (uint32)fd->size();
|
|
|
|
_midiData = (uint8 *)malloc(size);
|
|
|
|
|
|
|
|
if (_midiData) {
|
|
|
|
fd->read(_midiData, size);
|
|
|
|
|
|
|
|
syncVolume(); // FIXME: syncVolume calls setVolume which in turn also locks the mutex! ugh
|
|
|
|
|
|
|
|
_parser = MidiParser::createParser_SMF();
|
|
|
|
_parser->loadMusic(_midiData, size);
|
|
|
|
_parser->setTrack(0);
|
|
|
|
_parser->setMidiDriver(this);
|
|
|
|
_parser->setTimerRate(_driver->getBaseTempo());
|
2018-06-16 21:59:48 +10:00
|
|
|
_isLooping = true;
|
2018-05-01 20:33:10 +10:00
|
|
|
_isPlaying = true;
|
|
|
|
}
|
|
|
|
fd->close();
|
|
|
|
delete fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MidiPlayer::pause(bool p) {
|
|
|
|
_paused = p;
|
|
|
|
|
|
|
|
for (int i = 0; i < kNumChannels; ++i) {
|
|
|
|
if (_channelsTable[i]) {
|
|
|
|
_channelsTable[i]->volume(_paused ? 0 : _channelsVolume[i] * _masterVolume / 255);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void MidiPlayer::onTimer() {
|
|
|
|
Common::StackLock lock(_mutex);
|
|
|
|
|
|
|
|
if (!_paused && _isPlaying && _parser) {
|
|
|
|
_parser->onTimer();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void MidiPlayer::sendToChannel(byte channel, uint32 b) {
|
|
|
|
if (!_channelsTable[channel]) {
|
|
|
|
_channelsTable[channel] = (channel == 9) ? _driver->getPercussionChannel() : _driver->allocateChannel();
|
|
|
|
// If a new channel is allocated during the playback, make sure
|
|
|
|
// its volume is correctly initialized.
|
|
|
|
if (_channelsTable[channel])
|
|
|
|
_channelsTable[channel]->volume(_channelsVolume[channel] * _masterVolume / 255);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_channelsTable[channel])
|
|
|
|
_channelsTable[channel]->send(b);
|
|
|
|
}
|
|
|
|
|
2018-05-05 22:01:16 +10:00
|
|
|
void MidiPlayer::fade(int16 finalVolume, int16 duration) {
|
|
|
|
//TODO fade here.
|
|
|
|
debug(0, "Fade midi. finalVolume: %d, duration: %d", finalVolume, duration);
|
|
|
|
}
|
|
|
|
|
2014-04-18 23:12:10 +02:00
|
|
|
// VoicePlayer
|
|
|
|
|
2014-04-26 00:08:37 +02:00
|
|
|
VoicePlayer::VoicePlayer() {
|
2014-04-18 23:12:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
VoicePlayer::~VoicePlayer() {
|
|
|
|
stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool VoicePlayer::cue(const char *voiceName) {
|
2015-11-24 00:10:10 +01:00
|
|
|
debug(1, "VoicePlayer::cue(%s)", voiceName);
|
2014-04-18 23:12:10 +02:00
|
|
|
_voiceName = voiceName;
|
|
|
|
_voiceStatus = 2;
|
|
|
|
if (!isEnabled()) {
|
|
|
|
_voiceStatus = 3;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoicePlayer::stopCueing() {
|
|
|
|
_voiceStatus = 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoicePlayer::start(int16 volume, int16 pan) {
|
|
|
|
Common::String filename = Common::String::format("%s.wav", _voiceName.c_str());
|
|
|
|
Common::File *fd = new Common::File();
|
|
|
|
fd->open(filename);
|
|
|
|
Audio::AudioStream *audioStream = Audio::makeWAVStream(fd, DisposeAfterUse::YES);
|
2014-04-26 00:08:37 +02:00
|
|
|
g_system->getMixer()->playStream(Audio::Mixer::kSpeechSoundType, &_soundHandle, audioStream, -1, volume, pan);
|
2014-04-18 23:12:10 +02:00
|
|
|
_voiceStatus = 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
void VoicePlayer::stop() {
|
2014-04-26 00:08:37 +02:00
|
|
|
if (g_system->getMixer()->isSoundHandleActive(_soundHandle))
|
|
|
|
g_system->getMixer()->stopHandle(_soundHandle);
|
2014-04-18 23:12:10 +02:00
|
|
|
_voiceStatus = 1;
|
|
|
|
_voiceName.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool VoicePlayer::isPlaying() {
|
2014-04-26 00:08:37 +02:00
|
|
|
return g_system->getMixer()->isSoundHandleActive(_soundHandle);
|
2014-04-18 23:12:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool VoicePlayer::isEnabled() {
|
|
|
|
// TODO
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool VoicePlayer::isCued() {
|
|
|
|
return _voiceStatus == 2;
|
|
|
|
}
|
|
|
|
|
2014-04-26 00:08:37 +02:00
|
|
|
// Sound
|
|
|
|
|
|
|
|
Sound::Sound(uint32 soundEffectId, uint32 soundGroupId, bool looping)
|
|
|
|
: _stream(0), _soundEffectId(soundEffectId), _soundGroupId(soundGroupId), _looping(looping) {
|
|
|
|
load();
|
|
|
|
}
|
|
|
|
|
|
|
|
Sound::~Sound() {
|
|
|
|
unload();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Sound::load() {
|
|
|
|
Common::String filename = Common::String::format("%08x/%08x.wav", _soundGroupId, _soundEffectId);
|
|
|
|
Common::File *fd = new Common::File();
|
|
|
|
if (!fd->open(filename)) {
|
|
|
|
delete fd;
|
|
|
|
error("SoundMan::loadSound() Could not load %s", filename.c_str());
|
|
|
|
}
|
|
|
|
_stream = Audio::makeWAVStream(fd, DisposeAfterUse::YES);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Sound::unload() {
|
2015-11-24 00:10:10 +01:00
|
|
|
debug(1, "Sound::unload() %08X", _soundEffectId);
|
2014-04-26 00:08:37 +02:00
|
|
|
stop();
|
|
|
|
delete _stream;
|
|
|
|
_stream = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Sound::play(int16 volume, int16 pan) {
|
|
|
|
stop();
|
|
|
|
_stream->rewind();
|
|
|
|
Audio::AudioStream *audioStream = new Audio::LoopingAudioStream(_stream, _looping ? 0 : 1, DisposeAfterUse::NO);
|
|
|
|
g_system->getMixer()->playStream(Audio::Mixer::kSFXSoundType, &_soundHandle, audioStream,
|
|
|
|
-1, volume, pan, DisposeAfterUse::YES);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Sound::stop() {
|
2014-08-01 22:53:07 +02:00
|
|
|
if (isPlaying())
|
2014-04-26 00:08:37 +02:00
|
|
|
g_system->getMixer()->stopHandle(_soundHandle);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Sound::isPlaying() {
|
|
|
|
return g_system->getMixer()->isSoundHandleActive(_soundHandle);
|
|
|
|
}
|
|
|
|
|
2014-04-18 22:08:34 +02:00
|
|
|
// SoundMan
|
|
|
|
|
|
|
|
SoundMan::SoundMan(IllusionsEngine *vm)
|
|
|
|
: _vm(vm), _musicNotifyThreadId(0) {
|
2014-04-26 00:08:37 +02:00
|
|
|
_musicPlayer = new MusicPlayer();
|
2018-05-01 20:33:10 +10:00
|
|
|
_midiPlayer = new MidiPlayer();
|
2014-04-26 00:08:37 +02:00
|
|
|
_voicePlayer = new VoicePlayer();
|
2014-04-18 22:08:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
SoundMan::~SoundMan() {
|
|
|
|
delete _musicPlayer;
|
2018-05-01 20:33:10 +10:00
|
|
|
delete _midiPlayer;
|
2014-04-18 23:12:10 +02:00
|
|
|
delete _voicePlayer;
|
2014-04-26 00:08:37 +02:00
|
|
|
unloadSounds(0);
|
2014-04-18 22:08:34 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void SoundMan::update() {
|
|
|
|
// TODO voc_testCued();
|
|
|
|
if (_musicNotifyThreadId && !_musicPlayer->isPlaying())
|
|
|
|
_vm->notifyThreadId(_musicNotifyThreadId);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SoundMan::playMusic(uint32 musicId, int16 type, int16 volume, int16 pan, uint32 notifyThreadId) {
|
|
|
|
_vm->notifyThreadId(_musicNotifyThreadId);
|
|
|
|
_musicPlayer->play(musicId, type == 2, volume, pan);
|
|
|
|
_musicNotifyThreadId = notifyThreadId;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SoundMan::stopMusic() {
|
|
|
|
_musicPlayer->stop();
|
|
|
|
}
|
|
|
|
|
2014-04-18 23:12:10 +02:00
|
|
|
bool SoundMan::cueVoice(const char *voiceName) {
|
|
|
|
return _voicePlayer->cue(voiceName);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SoundMan::stopCueingVoice() {
|
|
|
|
_voicePlayer->stopCueing();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SoundMan::startVoice(int16 volume, int16 pan) {
|
2018-06-17 00:07:36 +10:00
|
|
|
_voicePlayer->start(calcAdjustedVolume("speech_volume", (uint8)volume), pan);
|
2014-04-18 23:12:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void SoundMan::stopVoice() {
|
|
|
|
_voicePlayer->stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SoundMan::isVoicePlaying() {
|
|
|
|
return _voicePlayer->isPlaying();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SoundMan::isVoiceEnabled() {
|
|
|
|
return _voicePlayer->isEnabled();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SoundMan::isVoiceCued() {
|
|
|
|
return _voicePlayer->isCued();
|
|
|
|
}
|
|
|
|
|
2014-04-26 00:08:37 +02:00
|
|
|
void SoundMan::loadSound(uint32 soundEffectId, uint32 soundGroupId, bool looping) {
|
2014-08-01 22:53:07 +02:00
|
|
|
Sound *sound = new Sound(soundEffectId, soundGroupId, looping);
|
|
|
|
_sounds.push_front(sound);
|
2014-04-26 00:08:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void SoundMan::playSound(uint32 soundEffectId, int16 volume, int16 pan) {
|
2014-08-01 22:53:07 +02:00
|
|
|
Sound *sound = getSound(soundEffectId);
|
|
|
|
if (sound)
|
2018-06-17 00:07:36 +10:00
|
|
|
sound->play(calcAdjustedVolume("sfx_volume", (uint8)volume), pan);
|
2014-04-26 00:08:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void SoundMan::stopSound(uint32 soundEffectId) {
|
2014-08-01 22:53:07 +02:00
|
|
|
Sound *sound = getSound(soundEffectId);
|
|
|
|
if (sound)
|
|
|
|
sound->stop();
|
2014-04-26 00:08:37 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void SoundMan::unloadSounds(uint32 soundGroupId) {
|
|
|
|
SoundListIterator it = _sounds.begin();
|
|
|
|
while (it != _sounds.end()) {
|
2014-08-01 22:53:07 +02:00
|
|
|
Sound *sound = *it;
|
|
|
|
if (soundGroupId == 0 || sound->_soundGroupId == soundGroupId) {
|
|
|
|
delete sound;
|
2014-04-26 00:08:37 +02:00
|
|
|
it = _sounds.erase(it);
|
|
|
|
} else
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Sound *SoundMan::getSound(uint32 soundEffectId) {
|
|
|
|
for (SoundListIterator it = _sounds.begin(); it != _sounds.end(); ++it)
|
|
|
|
if ((*it)->_soundEffectId == soundEffectId)
|
|
|
|
return *it;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-05-01 20:33:10 +10:00
|
|
|
void SoundMan::playMidiMusic(uint32 musicId) {
|
|
|
|
Common::String filename = Common::String::format("%08x.MID", musicId);
|
|
|
|
_midiPlayer->play(filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SoundMan::stopMidiMusic() {
|
|
|
|
_midiPlayer->stop();
|
|
|
|
}
|
|
|
|
|
2018-05-05 22:01:16 +10:00
|
|
|
void SoundMan::fadeMidiMusic(int16 finalVolume, int16 duration) {
|
|
|
|
_midiPlayer->fade(finalVolume, duration);
|
|
|
|
}
|
|
|
|
|
2018-06-19 22:33:09 +10:00
|
|
|
void SoundMan::setMusicVolume(uint16 volume) {
|
2018-06-17 00:07:36 +10:00
|
|
|
ConfMan.setInt("music_volume", volume);
|
|
|
|
_midiPlayer->syncVolume();
|
2018-06-18 22:13:25 +10:00
|
|
|
ConfMan.flushToDisk();
|
2018-06-17 00:07:36 +10:00
|
|
|
}
|
|
|
|
|
2018-06-19 22:33:09 +10:00
|
|
|
void SoundMan::setSfxVolume(uint16 volume) {
|
2018-06-17 00:07:36 +10:00
|
|
|
ConfMan.setInt("sfx_volume", volume);
|
2018-06-18 22:13:25 +10:00
|
|
|
ConfMan.flushToDisk();
|
2018-06-17 00:07:36 +10:00
|
|
|
}
|
|
|
|
|
2018-06-19 22:33:09 +10:00
|
|
|
void SoundMan::setSpeechVolume(uint16 volume) {
|
2018-06-17 00:07:36 +10:00
|
|
|
ConfMan.setInt("speech_volume", volume);
|
2018-06-18 22:13:25 +10:00
|
|
|
ConfMan.flushToDisk();
|
2018-06-17 00:07:36 +10:00
|
|
|
}
|
|
|
|
|
2018-06-19 22:33:09 +10:00
|
|
|
uint16 SoundMan::calcAdjustedVolume(const Common::String &volumeConfigKey, uint16 volume) {
|
|
|
|
uint16 masterVolume = (uint16)ConfMan.getInt(volumeConfigKey);
|
|
|
|
return (uint16)(((float)masterVolume/256) * (float)volume);
|
2018-06-17 00:07:36 +10:00
|
|
|
}
|
|
|
|
|
2018-06-19 22:33:09 +10:00
|
|
|
uint16 SoundMan::getMusicVolume() {
|
|
|
|
return (uint16)ConfMan.getInt("music_volume");
|
2018-06-17 00:07:36 +10:00
|
|
|
}
|
|
|
|
|
2018-06-19 22:33:09 +10:00
|
|
|
uint16 SoundMan::getSfxVolume() {
|
|
|
|
return (uint16)ConfMan.getInt("sfx_volume");
|
2018-06-17 00:07:36 +10:00
|
|
|
}
|
|
|
|
|
2018-06-19 22:33:09 +10:00
|
|
|
uint16 SoundMan::getSpeechVolume() {
|
|
|
|
return (uint16)ConfMan.getInt("speech_volume");
|
2018-06-17 00:07:36 +10:00
|
|
|
}
|
|
|
|
|
2014-04-18 22:08:34 +02:00
|
|
|
} // End of namespace Illusions
|