@ -30,6 +30,11 @@
# endif
# endif
namespace
{
const int sLoudnessFPS = 20 ; // loudness values per second of audio
}
namespace MWSound
namespace MWSound
{
{
SoundManager : : SoundManager ( const VFS : : Manager * vfs , bool useSound )
SoundManager : : SoundManager ( const VFS : : Manager * vfs , bool useSound )
@ -103,13 +108,6 @@ namespace MWSound
mOutput - > unloadSound ( sfxiter - > second . mHandle ) ;
mOutput - > unloadSound ( sfxiter - > second . mHandle ) ;
sfxiter - > second . mHandle = 0 ;
sfxiter - > second . mHandle = 0 ;
}
}
sfxiter = mVoiceSoundBuffers . begin ( ) ;
for ( ; sfxiter ! = mVoiceSoundBuffers . end ( ) ; + + sfxiter )
{
if ( sfxiter - > second . mHandle )
mOutput - > unloadSound ( sfxiter - > second . mHandle ) ;
sfxiter - > second . mHandle = 0 ;
}
}
}
mOutput . reset ( ) ;
mOutput . reset ( ) ;
}
}
@ -188,32 +186,37 @@ namespace MWSound
return sfx ;
return sfx ;
}
}
const Sound_Buffer * SoundManager : : lookup Voice( const std : : string & voicefile )
void SoundManager : : load Voice( const std : : string & voicefile )
{
{
NameBufferMap : : iterator sfxiter = mVoiceSoundBuffers . find ( voicefile ) ;
NameLoudnessMap : : iterator lipiter = mVoiceLipBuffers . find ( voicefile ) ;
if ( sfxiter = = mVoiceSoundBuffers . end ( ) )
if ( lipiter ! = mVoiceLipBuffers . end ( ) ) return ;
DecoderPtr decoder = getDecoder ( ) ;
// Workaround: Bethesda at some point converted some of the files to mp3, but the references were kept as .wav.
if ( decoder - > mResourceMgr - > exists ( voicefile ) )
decoder - > open ( voicefile ) ;
else
{
{
MWBase : : World * world = MWBase : : Environment : : get ( ) . getWorld ( ) ;
std : : string file = voicefile ;
static const float fAudioMinDistanceMult = world - > getStore ( ) . get < ESM : : GameSetting > ( ) . find ( " fAudioMinDistanceMult " ) - > getFloat ( ) ;
std : : string : : size_type pos = file . rfind ( ' . ' ) ;
static const float fAudioMaxDistanceMult = world - > getStore ( ) . get < ESM : : GameSetting > ( ) . find ( " fAudioMaxDistanceMult " ) - > getFloat ( ) ;
if ( pos ! = std : : string : : npos )
static const float fAudioVoiceDefaultMinDistance = world - > getStore ( ) . get < ESM : : GameSetting > ( ) . find ( " fAudioVoiceDefaultMinDistance " ) - > getFloat ( ) ;
file = file . substr ( 0 , pos ) + " .mp3 " ;
static const float fAudioVoiceDefaultMaxDistance = world - > getStore ( ) . get < ESM : : GameSetting > ( ) . find ( " fAudioVoiceDefaultMaxDistance " ) - > getFloat ( ) ;
decoder - > open ( file ) ;
}
float minDistance = fAudioVoiceDefaultMinDistance * fAudioMinDistanceMult ;
ChannelConfig chans ;
float maxDistance = fAudioVoiceDefaultMaxDistance * fAudioMaxDistanceMult ;
SampleType type ;
minDistance = std : : max ( minDistance , 1.f ) ;
int srate ;
maxDistance = std : : max ( minDistance , maxDistanc e) ;
decoder - > getInfo ( & srate , & chans , & typ e) ;
sfxiter = mVoiceSoundBuffers . insert ( std : : make_pair (
std : : vector < char > data ;
voicefile , Sound_Buffer ( " sound/ " + voicefile , 1.0f , minDistance , maxDistance )
decoder - > readAll ( data ) ;
) ) . first ;
decoder - > close ( ) ;
mVFS - > normalizeFilename ( sfxiter - > second . mResourceName ) ;
sfxiter - > second . mHandle = mOutput - > loadSound ( sfxiter - > second . mResourceName , & sfxiter - > second . mLoudness ) ;
}
else if ( ! sfxiter - > second . mHandle )
sfxiter - > second . mHandle = mOutput - > loadSound ( sfxiter - > second . mResourceName , & sfxiter - > second . mLoudness ) ;
return & sfxiter - > second ;
Sound_Loudness loudness ;
loudness . analyzeLoudness ( data , srate , chans , type , static_cast < float > ( sLoudnessFPS ) ) ;
mVoiceLipBuffers . insert ( std : : make_pair ( voicefile , loudness ) ) ;
}
}
@ -336,14 +339,25 @@ namespace MWSound
return ;
return ;
try
try
{
{
std : : string voicefile = Misc : : StringUtils : : lowerCase ( filename ) ;
MWBase : : World * world = MWBase : : Environment : : get ( ) . getWorld ( ) ;
const Sound_Buffer * sfx = lookupVoice ( voicefile ) ;
static const float fAudioMinDistanceMult = world - > getStore ( ) . get < ESM : : GameSetting > ( ) . find ( " fAudioMinDistanceMult " ) - > getFloat ( ) ;
static const float fAudioMaxDistanceMult = world - > getStore ( ) . get < ESM : : GameSetting > ( ) . find ( " fAudioMaxDistanceMult " ) - > getFloat ( ) ;
static const float fAudioVoiceDefaultMinDistance = world - > getStore ( ) . get < ESM : : GameSetting > ( ) . find ( " fAudioVoiceDefaultMinDistance " ) - > getFloat ( ) ;
static const float fAudioVoiceDefaultMaxDistance = world - > getStore ( ) . get < ESM : : GameSetting > ( ) . find ( " fAudioVoiceDefaultMaxDistance " ) - > getFloat ( ) ;
static float minDistance = std : : max ( fAudioVoiceDefaultMinDistance * fAudioMinDistanceMult , 1.0f ) ;
static float maxDistance = std : : max ( fAudioVoiceDefaultMaxDistance * fAudioMaxDistanceMult , minDistance ) ;
std : : string voicefile = " sound/ " + Misc : : StringUtils : : lowerCase ( filename ) ;
float basevol = volumeFromType ( Play_TypeVoice ) ;
float basevol = volumeFromType ( Play_TypeVoice ) ;
const ESM : : Position & pos = ptr . getRefData ( ) . getPosition ( ) ;
const ESM : : Position & pos = ptr . getRefData ( ) . getPosition ( ) ;
const osg : : Vec3f objpos ( pos . asVec3 ( ) ) ;
const osg : : Vec3f objpos ( pos . asVec3 ( ) ) ;
MWBase : : SoundPtr sound = mOutput - > playSound3D ( sfx - > mHandle ,
loadVoice ( voicefile ) ;
objpos , sfx - > mVolume , basevol , 1.0f , sfx - > mMinDist , sfx - > mMaxDist , Play_Normal | Play_TypeVoice , 0
DecoderPtr decoder = getDecoder ( ) ;
decoder - > open ( voicefile ) ;
MWBase : : SoundPtr sound = mOutput - > streamSound3D ( decoder ,
objpos , 1.0f , basevol , 1.0f , minDistance , maxDistance , Play_Normal | Play_TypeVoice
) ;
) ;
mActiveSaySounds [ ptr ] = std : : make_pair ( sound , voicefile ) ;
mActiveSaySounds [ ptr ] = std : : make_pair ( sound , voicefile ) ;
}
}
@ -358,12 +372,13 @@ namespace MWSound
SaySoundMap : : const_iterator snditer = mActiveSaySounds . find ( ptr ) ;
SaySoundMap : : const_iterator snditer = mActiveSaySounds . find ( ptr ) ;
if ( snditer ! = mActiveSaySounds . end ( ) )
if ( snditer ! = mActiveSaySounds . end ( ) )
{
{
NameBufferMap : : const_iterator sfxiter = mVoiceSoundBuffers . find ( snditer - > second . second ) ;
MWBase : : SoundPtr sound = snditer - > second . first ;
if ( sfxiter ! = mVoiceSoundBuffers . end ( ) )
NameLoudnessMap : : const_iterator lipiter = mVoiceLipBuffers . find ( snditer - > second . second ) ;
if ( lipiter ! = mVoiceLipBuffers . end ( ) )
{
{
float sec = s nditer- > sec ond. first - > getTimeOffset ( ) ;
float sec = s ou nd- > getTimeOffset ( ) ;
if ( s nditer- > sec ond. first - > isPlaying ( ) )
if ( s ou nd- > isPlaying ( ) )
return sfxiter- > second . mLoudness . getLoudnessAtTime ( sec ) ;
return lipiter- > second . getLoudnessAtTime ( sec ) ;
}
}
}
}
@ -376,12 +391,15 @@ namespace MWSound
return ;
return ;
try
try
{
{
std : : string voicefile = Misc : : StringUtils : : lowerCase ( filename ) ;
std : : string voicefile = " sound/ " + Misc : : StringUtils : : lowerCase ( filename ) ;
const Sound_Buffer * sfx = lookupVoice ( voicefile ) ;
float basevol = volumeFromType ( Play_TypeVoice ) ;
float basevol = volumeFromType ( Play_TypeVoice ) ;
MWBase : : SoundPtr sound = mOutput - > playSound ( sfx - > mHandle ,
loadVoice ( voicefile ) ;
sfx - > mVolume , basevol , 1.0f , Play_Normal | Play_TypeVoice , 0
DecoderPtr decoder = getDecoder ( ) ;
decoder - > open ( voicefile ) ;
MWBase : : SoundPtr sound = mOutput - > streamSound ( decoder ,
basevol , 1.0f , Play_Normal | Play_TypeVoice
) ;
) ;
mActiveSaySounds [ MWWorld : : Ptr ( ) ] = std : : make_pair ( sound , voicefile ) ;
mActiveSaySounds [ MWWorld : : Ptr ( ) ] = std : : make_pair ( sound , voicefile ) ;
}
}