mirror of
https://github.com/libretro/scummvm.git
synced 2025-02-23 04:33:09 +00:00

This avoids having the volume reset on game startup. Also, volume is no longer stored in saved games
313 lines
8.5 KiB
C++
313 lines
8.5 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 "audio/audiostream.h"
|
|
#include "audio/mixer.h"
|
|
#include "audio/decoders/raw.h"
|
|
#include "audio/mods/protracker.h"
|
|
#include "common/config-manager.h"
|
|
#include "common/memstream.h"
|
|
#include "common/system.h"
|
|
#include "chewy/resource.h"
|
|
#include "chewy/sound.h"
|
|
#include "chewy/types.h"
|
|
#include "chewy/globals.h"
|
|
#include "chewy/audio/chewy_voc.h"
|
|
#include "chewy/audio/tmf_stream.h"
|
|
|
|
namespace Chewy {
|
|
|
|
Sound::Sound(Audio::Mixer *mixer) {
|
|
_mixer = mixer;
|
|
_speechRes = new SoundResource("speech.tvp");
|
|
_soundRes = new SoundResource("details.tap");
|
|
}
|
|
|
|
Sound::~Sound() {
|
|
delete _soundRes;
|
|
delete _speechRes;
|
|
}
|
|
|
|
void Sound::playSound(int num, uint channel, bool loop) {
|
|
SoundChunk *sound = _soundRes->getSound(num);
|
|
uint8 *data = (uint8 *)MALLOC(sound->size);
|
|
memcpy(data, sound->data, sound->size);
|
|
|
|
playSound(data, sound->size, channel, loop);
|
|
|
|
delete[] sound->data;
|
|
delete sound;
|
|
}
|
|
|
|
void Sound::playSound(uint8 *data, uint32 size, uint channel, bool loop, DisposeAfterUse::Flag dispose) {
|
|
stopSound(channel);
|
|
|
|
Audio::AudioStream *stream = Audio::makeLoopingAudioStream(
|
|
new ChewyVocStream(
|
|
new Common::MemorySeekableReadWriteStream(data, size, dispose),
|
|
dispose),
|
|
loop ? 0 : 1);
|
|
|
|
_mixer->playStream(Audio::Mixer::kSFXSoundType, &_soundHandle[channel], stream);
|
|
}
|
|
|
|
void Sound::pauseSound(uint channel) {
|
|
assert(channel < MAX_SOUND_EFFECTS);
|
|
_mixer->pauseHandle(_soundHandle[channel], true);
|
|
}
|
|
|
|
void Sound::resumeSound(uint channel) {
|
|
assert(channel < MAX_SOUND_EFFECTS);
|
|
_mixer->pauseHandle(_soundHandle[channel], false);
|
|
}
|
|
|
|
void Sound::stopSound(uint channel) {
|
|
assert(channel < MAX_SOUND_EFFECTS);
|
|
_mixer->stopHandle(_soundHandle[channel]);
|
|
}
|
|
|
|
void Sound::stopAllSounds() {
|
|
for (int i = 0; i < 14; i++)
|
|
stopSound(i);
|
|
}
|
|
|
|
bool Sound::isSoundActive(uint channel) const {
|
|
assert(channel < MAX_SOUND_EFFECTS);
|
|
return _mixer->isSoundHandleActive(_soundHandle[channel]);
|
|
}
|
|
|
|
void Sound::setSoundVolume(uint volume) {
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kSFXSoundType, volume);
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kSpeechSoundType, volume);
|
|
}
|
|
|
|
int Sound::getSoundVolume() const {
|
|
return _mixer->getVolumeForSoundType(Audio::Mixer::kSFXSoundType);
|
|
}
|
|
|
|
void Sound::pushVolume() {
|
|
_soundVolume = _mixer->getVolumeForSoundType(Audio::Mixer::kSFXSoundType);
|
|
_speechVolume = _mixer->getVolumeForSoundType(Audio::Mixer::kSpeechSoundType);
|
|
_musicVolume = _mixer->getVolumeForSoundType(Audio::Mixer::kMusicSoundType);
|
|
}
|
|
|
|
void Sound::popVolume() {
|
|
assert(_soundVolume >= 0 && _speechVolume >= 0 && _musicVolume >= 0);
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kSFXSoundType, _soundVolume);
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kSpeechSoundType, _speechVolume);
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kMusicSoundType, _musicVolume);
|
|
}
|
|
|
|
void Sound::setSoundChannelVolume(uint channel, uint volume) {
|
|
assert(channel < MAX_SOUND_EFFECTS);
|
|
_mixer->setChannelVolume(_soundHandle[channel], volume);
|
|
}
|
|
|
|
void Sound::setSoundChannelBalance(uint channel, int8 balance) {
|
|
assert(channel < MAX_SOUND_EFFECTS);
|
|
_mixer->setChannelBalance(_soundHandle[channel], balance);
|
|
}
|
|
|
|
void Sound::playMusic(int16 num, bool loop) {
|
|
uint32 musicNum = _soundRes->getChunkCount() - 1 - num;
|
|
Chunk *chunk = _soundRes->getChunk(musicNum);
|
|
uint8 *data = _soundRes->getChunkData(musicNum);
|
|
_curMusic = num;
|
|
|
|
playMusic(data, chunk->size);
|
|
|
|
delete[] data;
|
|
}
|
|
|
|
void Sound::playMusic(uint8 *data, uint32 size) {
|
|
TMFStream *stream = new TMFStream(new Common::MemoryReadStream(data, size), 0);
|
|
_curMusic = -1;
|
|
|
|
_mixer->playStream(Audio::Mixer::kMusicSoundType, &_musicHandle, stream);
|
|
}
|
|
|
|
void Sound::pauseMusic() {
|
|
_mixer->pauseHandle(_musicHandle, true);
|
|
}
|
|
|
|
void Sound::resumeMusic() {
|
|
_mixer->pauseHandle(_musicHandle, false);
|
|
}
|
|
|
|
void Sound::stopMusic() {
|
|
_curMusic = -1;
|
|
_mixer->stopHandle(_musicHandle);
|
|
}
|
|
|
|
bool Sound::isMusicActive() const {
|
|
return _mixer->isSoundHandleActive(_musicHandle);
|
|
}
|
|
|
|
void Sound::setMusicVolume(uint volume) {
|
|
_mixer->setVolumeForSoundType(Audio::Mixer::kMusicSoundType, volume);
|
|
}
|
|
|
|
int Sound::getMusicVolume() const {
|
|
return _mixer->getVolumeForSoundType(Audio::Mixer::kMusicSoundType);
|
|
}
|
|
|
|
void Sound::playSpeech(int num, bool waitForFinish) {
|
|
if (isSpeechActive())
|
|
stopSpeech();
|
|
|
|
// Get the speech data
|
|
SoundChunk *sound = _speechRes->getSound(num);
|
|
size_t size = sound->size;
|
|
uint8 *data = (uint8 *)MALLOC(size);
|
|
memcpy(data, sound->data, size);
|
|
|
|
delete[] sound->data;
|
|
delete sound;
|
|
|
|
// Play it
|
|
Audio::AudioStream *stream = new ChewyVocStream(
|
|
new Common::MemorySeekableReadWriteStream(data, size, DisposeAfterUse::YES),
|
|
DisposeAfterUse::YES);
|
|
|
|
_mixer->playStream(Audio::Mixer::kSpeechSoundType,
|
|
&_speechHandle, stream);
|
|
|
|
if (waitForFinish) {
|
|
// Wait for speech to finish
|
|
while (isSpeechActive() && !SHOULD_QUIT) {
|
|
setupScreen(DO_SETUP);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Sound::pauseSpeech() {
|
|
_mixer->pauseHandle(_speechHandle, true);
|
|
}
|
|
|
|
void Sound::resumeSpeech() {
|
|
_mixer->pauseHandle(_speechHandle, false);
|
|
}
|
|
|
|
void Sound::stopSpeech() {
|
|
_mixer->stopHandle(_speechHandle);
|
|
}
|
|
|
|
bool Sound::isSpeechActive() const {
|
|
return _mixer->isSoundHandleActive(_speechHandle);
|
|
}
|
|
|
|
void Sound::stopAll() {
|
|
_mixer->stopAll();
|
|
}
|
|
|
|
void Sound::waitForSpeechToFinish() {
|
|
if (speechEnabled()) {
|
|
while (isSpeechActive() && !SHOULD_QUIT) {
|
|
setupScreen(DO_SETUP);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool Sound::soundEnabled() const {
|
|
return !ConfMan.getBool("sfx_mute");
|
|
}
|
|
|
|
void Sound::toggleSound(bool enable) {
|
|
return ConfMan.setBool("sfx_mute", !enable);
|
|
}
|
|
|
|
bool Sound::musicEnabled() const {
|
|
return !ConfMan.getBool("music_mute");
|
|
}
|
|
|
|
void Sound::toggleMusic(bool enable) {
|
|
return ConfMan.setBool("music_mute", !enable);
|
|
}
|
|
|
|
bool Sound::speechEnabled() const {
|
|
return !ConfMan.getBool("speech_mute");
|
|
}
|
|
|
|
void Sound::toggleSpeech(bool enable) {
|
|
return ConfMan.setBool("speech_mute", !enable);
|
|
}
|
|
|
|
bool Sound::subtitlesEnabled() const {
|
|
return ConfMan.getBool("subtitles");
|
|
}
|
|
|
|
void Sound::toggleSubtitles(bool enable) {
|
|
return ConfMan.setBool("subtitles", enable);
|
|
}
|
|
|
|
struct RoomMusic {
|
|
int16 room;
|
|
int16 music;
|
|
};
|
|
|
|
const RoomMusic roomMusic[] = {
|
|
{ 0, 13 }, { 1, 17 }, { 18, 17 }, { 90, 17 }, { 2, 43 },
|
|
{ 88, 43 }, { 3, 0 }, { 4, 0 }, { 5, 14 }, { 8, 14 },
|
|
{ 12, 14 }, { 86, 14 }, { 6, 1 }, { 7, 18 }, { 97, 18 },
|
|
{ 9, 20 }, { 10, 20 }, { 47, 20 }, { 87, 20 }, { 11, 19 },
|
|
{ 14, 19 }, { 15, 16 }, { 16, 16 }, { 19, 16 }, { 96, 16 },
|
|
{ 21, 2 }, { 22, 48 }, { 25, 11 }, { 26, 11 }, { 27, 33 },
|
|
{ 30, 33 }, { 54, 33 }, { 63, 33 }, { 28, 47 }, { 29, 47 },
|
|
{ 31, 9 }, { 35, 9 }, { 32, 38 }, { 40, 38 }, { 71, 38 },
|
|
{ 89, 38 }, { 92, 38 }, { 33, 35 }, { 37, 8 }, { 39, 9 },
|
|
{ 42, 41 }, { 45, 44 }, { 46, 21 }, { 50, 21 }, { 73, 21 },
|
|
{ 74, 21 }, { 48, 22 }, { 49, 3 }, { 51, 27 }, { 52, 27 },
|
|
{ 53, 26 }, { 55, 23 }, { 57, 23 }, { 56, 52 }, { 62, 25 },
|
|
{ 64, 51 }, { 66, 34 }, { 68, 34 }, { 67, 28 }, { 69, 28 },
|
|
{ 70, 28 }, { 75, 28 }, { 72, 31 }, { 76, 46 }, { 79, 6 },
|
|
{ 80, 29 }, { 81, 45 }, { 82, 50 }, { 84, 24 }, { 85, 32 },
|
|
{ 91, 36 }, { 94, 40 }, { 95, 40 }, { 98, 4 }, { 255, 5 },
|
|
{ 256, 10 }, { 257, 52 }, { 258, 53 }, { 259, 54 }, { 260, 24 },
|
|
{ -1, -1 }
|
|
};
|
|
|
|
void Sound::playRoomMusic(int16 roomNum) {
|
|
int16 musicIndex = -1;
|
|
if (!musicEnabled())
|
|
return;
|
|
|
|
for (const RoomMusic *cur = roomMusic; cur->room >= 0; ++cur) {
|
|
if (cur->room == roomNum) {
|
|
musicIndex = cur->music;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// TODO: Extra checks for two flags in room 56
|
|
//if ((spieler.flags32 & SpielerFlags32_10) != 0 && spieler.flags33 >= 0)
|
|
// musicIndex = 52;
|
|
//else
|
|
// musicIndex = 7;
|
|
|
|
if (musicIndex != _curMusic) {
|
|
stopMusic();
|
|
if (musicIndex >= 0)
|
|
playMusic(musicIndex, true);
|
|
}
|
|
}
|
|
|
|
} // namespace Chewy
|