scummvm/engines/director/castmember/sound.cpp
2025-02-04 15:24:17 +01:00

202 lines
5.7 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 "director/director.h"
#include "director/cast.h"
#include "director/sound.h"
#include "director/castmember/sound.h"
#include "director/lingo/lingo-the.h"
namespace Director {
SoundCastMember::SoundCastMember(Cast *cast, uint16 castId, Common::SeekableReadStreamEndian &stream, uint16 version)
: CastMember(cast, castId, stream) {
_type = kCastSound;
_audio = nullptr;
_looping = 0;
}
SoundCastMember::SoundCastMember(Cast *cast, uint16 castId, SoundCastMember &source)
: CastMember(cast, castId) {
_type = kCastSound;
_loaded = false;
_audio = nullptr;
_looping = source._looping;
warning("SoundCastMember(): Duplicating source %d to target %d! This is unlikely to work properly, as the resource loader is based on the cast ID", source._castId, castId);
}
SoundCastMember::~SoundCastMember() {
if (_audio)
delete _audio;
}
Common::String SoundCastMember::formatInfo() {
return Common::String::format(
"looping: %d", _looping
);
}
void SoundCastMember::load() {
if (_loaded)
return;
uint32 tag = 0;
uint16 sndId = 0;
if (_cast->_version < kFileVer400) {
tag = MKTAG('S', 'N', 'D', ' ');
sndId = (uint16)(_castId + _cast->_castIDoffset);
} else if (_cast->_version >= kFileVer400 && _cast->_version < kFileVer600) {
for (auto &it : _children) {
if (it.tag == MKTAG('s', 'n', 'd', ' ') || it.tag == MKTAG('S', 'N', 'D', ' ')) {
sndId = it.index;
tag = it.tag;
break;
}
}
if (!sndId) {
warning("SoundCastMember::load(): No snd resource found in %d children, falling back to D3", _children.size());
tag = MKTAG('S', 'N', 'D', ' ');
sndId = (uint16)(_castId + _cast->_castIDoffset);
}
} else {
warning("STUB: SoundCastMember::SoundCastMember(): Sounds not yet supported for version %d", _cast->_version);
}
Common::SeekableReadStreamEndian *sndData = _cast->getResource(tag, sndId);
if (!sndData) {
tag = MKTAG('s', 'n', 'd', ' ');
sndData = _cast->getResource(tag, sndId);
}
if (sndData == nullptr || sndData->size() == 0) {
// audio file is linked, load from the filesystem
Common::String res = _cast->getLinkedPath(_castId);
if (!res.empty()) {
debugC(2, kDebugLoading, "****** Loading file '%s', cast id: %d", res.c_str(), sndId);
AudioFileDecoder *audio = new AudioFileDecoder(res);
_audio = audio;
// Linked sound files always have the loop flag disabled
_looping = 0;
} else {
warning("Sound::load(): no resource or info found for cast member %d, skipping", _castId);
}
} else {
debugC(2, kDebugLoading, "****** Loading '%s' id: %d, %d bytes", tag2str(tag), sndId, (int)sndData->size());
SNDDecoder *audio = new SNDDecoder();
audio->loadStream(*sndData);
_audio = audio;
_size = sndData->size();
if (_cast->_version < kFileVer400) {
// The looping flag wasn't added to sound cast members until D4.
// In older versions, always loop sounds that contain a loop start and end.
_looping = audio->hasLoopBounds();
} else {
// Some sound cast members at version kFileVer400 have looping=true with
// invalid loop bounds (bigger than sample size or non-consecutive).
// Resetting loop bounds to sample bounds and disabling looping similar
// to how D4 playback seems to work.
if (!audio->hasValidLoopBounds()) {
// only emit a warning for files > kFileVer400 as it's only kFileVer400 files that should be affected
if (_cast->_version > kFileVer400) {
warning("Sound::load(): Invalid loop bounds detected. Disabling looping for cast member id %d, sndId %d", _castId, sndId);
} else {
debugC(2, "Sound::load(): Invalid loop bounds detected. Disabling looping for cast member id %d, sndId %d", _castId, sndId);
}
_looping = false;
audio->resetLoopBounds();
}
}
}
if (sndData)
delete sndData;
_loaded = true;
}
void SoundCastMember::unload() {
if (!_loaded)
return;
delete _audio;
_audio = nullptr;
_size = 0;
_looping = false;
_loaded = false;
}
bool SoundCastMember::hasField(int field) {
switch (field) {
case kTheChannelCount:
case kTheSampleRate:
case kTheSampleSize:
return true;
default:
break;
}
return CastMember::hasField(field);
}
Datum SoundCastMember::getField(int field) {
Datum d;
load();
if (!_audio) {
warning("SoundCastMember::getField(): Audio not found");
return d;
}
switch (field) {
case kTheChannelCount:
d = _audio->getChannelCount();
break;
case kTheSampleRate:
d = _audio->getSampleRate();
break;
case kTheSampleSize:
d = _audio->getSampleSize();
break;
default:
d = CastMember::getField(field);
}
return d;
}
bool SoundCastMember::setField(int field, const Datum &d) {
switch (field) {
case kTheChannelCount:
case kTheSampleRate:
case kTheSampleSize:
warning("SoundCastMember::setField(): Attempt to set read-only field %s of cast %d", g_lingo->field2str(field), _castId);
return false;
default:
break;
}
return CastMember::setField(field, d);
}
} // End of namespace Director