scummvm/engines/wage/sound.cpp
2023-09-10 21:38:58 +02:00

176 lines
5.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/>.
*
* MIT License:
*
* Copyright (c) 2009 Alexei Svitkine, Eugene Sandulenko
*
* Permission is hereby granted, free of charge, to any person
* obtaining a copy of this software and associated documentation
* files (the "Software"), to deal in the Software without
* restriction, including without limitation the rights to use,
* copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following
* conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
* OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
* OTHER DEALINGS IN THE SOFTWARE.
*
*/
#include "audio/audiostream.h"
#include "audio/mixer.h"
#include "audio/decoders/raw.h"
#include "common/stream.h"
#include "common/system.h"
#include "wage/wage.h"
#include "wage/entities.h"
#include "wage/sound.h"
#include "wage/world.h"
namespace Wage {
static const int8 deltas[] = { 0,-49,-36,-25,-16,-9,-4,-1,0,1,4,9,16,25,36,49 };
Sound::Sound(Common::String name, Common::SeekableReadStream *data) : _name(name) {
_size = data->size() - 20;
_data = (byte *)calloc(2 * _size, 1);
data->skip(20); // Skip header
byte value = 0x80;
for (uint i = 0; i < _size; i++) {
byte d = data->readByte();
value += deltas[d & 0xf];
_data[i * 2] = value;
value += deltas[(d >> 4) & 0xf];
_data[i * 2 + 1] = value;
}
}
Sound::~Sound() {
free(_data);
}
void WageEngine::playSound(Common::String soundName) {
soundName.toLowercase();
if (!_world->_sounds.contains(soundName)) {
warning("playSound: Sound '%s' does not exist", soundName.c_str());
return;
}
Sound *s = _world->_sounds[soundName];
Audio::AudioStream *stream = Audio::makeRawStream(s->_data, s->_size, 11000, Audio::FLAG_UNSIGNED);
_mixer->playStream(Audio::Mixer::kPlainSoundType, &_soundHandle, stream,
-1, Audio::Mixer::kMaxChannelVolume, 0, DisposeAfterUse::NO);
while (_mixer->isSoundHandleActive(_soundHandle) && !_shouldQuit) {
Common::Event event;
if (_eventMan->pollEvent(event)) {
switch (event.type) {
case Common::EVENT_QUIT:
if (saveDialog())
_shouldQuit = true;
break;
default:
break;
}
}
_system->updateScreen();
_system->delayMillis(10);
}
}
static void soundTimer(void *refCon) {
Scene *scene = (Scene *)refCon;
WageEngine *engine = (WageEngine *)g_engine;
g_system->getTimerManager()->removeTimerProc(&soundTimer);
if (engine->_world->_player->_currentScene != scene)
return;
if (engine->_soundQueue.empty()) {
if (scene->_soundType == Scene::PERIODIC && 0) {
engine->_soundToPlay = scene->_soundName; // We cannot play sound here because that goes recursively
uint32 nextRun = 60000 / scene->_soundFrequency;
g_system->getTimerManager()->installTimerProc(&soundTimer, nextRun * 1000, scene, "WageEngine::soundTimer");
} else if (scene->_soundType == Scene::RANDOM || 1) {
for (int i = 0; i < scene->_soundFrequency * 5; i++)
engine->_soundQueue.push_back(g_system->getMillis() + engine->_rnd->getRandomNumber(60000));
Common::sort(engine->_soundQueue.begin(), engine->_soundQueue.end());
int nextRun = engine->_soundQueue.front();
engine->_soundQueue.pop_front();
g_system->getTimerManager()->installTimerProc(&soundTimer, (nextRun - g_system->getMillis()) * 1000, scene, "WageEngine::soundTimer");
} else {
warning("updateSoundTimerForScene: Unknown sound type %d", scene->_soundType);
}
} else {
int nextRun = engine->_soundQueue.front();
engine->_soundQueue.pop_front();
g_system->getTimerManager()->installTimerProc(&soundTimer, (nextRun - g_system->getMillis()) * 1000, scene, "WageEngine::soundTimer");
engine->_soundToPlay = scene->_soundName; // We cannot play sound here because that goes recursively
}
}
void WageEngine::updateSoundTimerForScene(Scene *scene, bool firstTime) {
if (_world->_player->_currentScene != scene)
return;
if (scene->_soundFrequency > 0 && !scene->_soundName.empty()) {
Common::String soundName(scene->_soundName);
soundName.toLowercase();
if (!_world->_sounds.contains(soundName)) {
warning("updateSoundTimerForScene: Sound '%s' does not exist", soundName.c_str());
return;
}
// Launch the process
soundTimer(scene);
}
}
} // End of namespace Wage