forked from teamnwah/openmw-tes3coop
33d11a733c
git-svn-id: https://openmw.svn.sourceforge.net/svnroot/openmw/trunk@31 ea6a568a-9f4f-0410-981a-c910a81bb256
405 lines
9.7 KiB
D
405 lines
9.7 KiB
D
/*
|
|
OpenMW - The completely unofficial reimplementation of Morrowind
|
|
Copyright (C) 2008 Nicolay Korslund
|
|
Email: < korslund@gmail.com >
|
|
WWW: http://openmw.snaptoad.com/
|
|
|
|
This file (music.d) is part of the OpenMW package.
|
|
|
|
OpenMW is distributed as free software: you can redistribute it
|
|
and/or modify it under the terms of the GNU General Public License
|
|
version 3, as published by the Free Software Foundation.
|
|
|
|
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
|
|
version 3 along with this program. If not, see
|
|
http://www.gnu.org/licenses/ .
|
|
|
|
*/
|
|
|
|
module sound.music;
|
|
|
|
import sound.avcodec;
|
|
import sound.audio;
|
|
import sound.al;
|
|
|
|
import std.stdio;
|
|
import std.string;
|
|
|
|
import core.config;
|
|
import core.resource;
|
|
|
|
// Simple music player, has a playlist and can pause/resume music.
|
|
struct MusicManager
|
|
{
|
|
private:
|
|
|
|
// How much to add to the volume each second when fading
|
|
const float fadeInRate = 0.10;
|
|
const float fadeOutRate = 0.20;
|
|
|
|
// Maximum buffer length, divided up among OpenAL buffers
|
|
const uint bufLength = 128*1024;
|
|
|
|
// Volume
|
|
ALfloat volume, maxVolume;
|
|
|
|
char[] name;
|
|
|
|
void fail(char[] msg)
|
|
{
|
|
throw new SoundException(name ~ " Jukebox", msg);
|
|
}
|
|
|
|
// List of songs to play
|
|
char[][] playlist;
|
|
int index; // Index of next song to play
|
|
|
|
bool musicOn;
|
|
ALuint sID;
|
|
ALuint bIDs[4];
|
|
|
|
ALenum bufFormat;
|
|
ALint bufRate;
|
|
|
|
AVFile fileHandle;
|
|
AVAudio audioHandle;
|
|
ubyte[] outData;
|
|
|
|
// Which direction are we currently fading, if any
|
|
enum Fade { None = 0, In, Out }
|
|
Fade fading;
|
|
|
|
public:
|
|
|
|
// Initialize the jukebox
|
|
void initialize(char[] name)
|
|
{
|
|
this.name = name;
|
|
sID = 0;
|
|
foreach(ref b; bIDs) b = 0;
|
|
outData.length = bufLength / bIDs.length;
|
|
fileHandle = null;
|
|
musicOn = false;
|
|
}
|
|
|
|
// Get the new volume setting.
|
|
void updateVolume()
|
|
{
|
|
maxVolume = config.calcMusicVolume();
|
|
|
|
if(!musicOn) return;
|
|
|
|
// Adjust volume up to new setting, unless we are in the middle of
|
|
// a fade. Even if we are fading, though, the volume should never
|
|
// be over the max.
|
|
if(fading == Fade.None || volume > maxVolume) volume = maxVolume;
|
|
if(sID)
|
|
alSourcef(sID, AL_GAIN, volume);
|
|
}
|
|
|
|
// Give a music play list
|
|
void setPlaylist(char[][] pl)
|
|
{
|
|
playlist = pl;
|
|
index = 0;
|
|
|
|
randomize();
|
|
}
|
|
|
|
// Randomize playlist. If the argument is true, then we don't want
|
|
// the old last to be the new first.
|
|
private void randomize(bool checklast = false)
|
|
{
|
|
if(playlist.length < 2) return;
|
|
|
|
// Get the index of the last song played
|
|
int lastidx = ((index==0) ? (playlist.length-1) : (index-1));
|
|
|
|
foreach(int i, char[] s; playlist)
|
|
{
|
|
int idx = rnd.randInt(i,playlist.length-1);
|
|
|
|
// Don't put the last idx as the first entry
|
|
if(i == 0 && checklast && lastidx == idx)
|
|
{
|
|
idx++;
|
|
if(idx == playlist.length)
|
|
idx = i;
|
|
}
|
|
if(idx == i) /* skip if swapping with self */
|
|
continue;
|
|
playlist[i] = playlist[idx];
|
|
playlist[idx] = s;
|
|
}
|
|
}
|
|
|
|
// Skip to the next track
|
|
void playNext()
|
|
{
|
|
// If music is disabled, do nothing
|
|
if(!musicOn) return;
|
|
|
|
// No tracks to play?
|
|
if(!playlist.length) return;
|
|
|
|
// Generate a source to play back with if needed
|
|
if(!sID)
|
|
{
|
|
alGenSources(1, &sID);
|
|
if(checkALError())
|
|
fail("Couldn't generate music sources");
|
|
alSourcei(sID, AL_SOURCE_RELATIVE, AL_TRUE);
|
|
|
|
updateVolume();
|
|
}
|
|
else
|
|
{
|
|
// Kill current track
|
|
alSourceStop(sID);
|
|
alSourcei(sID, AL_BUFFER, 0);
|
|
alDeleteBuffers(bIDs.length, bIDs.ptr);
|
|
bIDs[] = 0;
|
|
checkALError("killing current track");
|
|
}
|
|
|
|
if(fileHandle) cpp_closeAVFile(fileHandle);
|
|
fileHandle = null;
|
|
audioHandle = null;
|
|
|
|
// End of list? Randomize and start over
|
|
if(index == playlist.length)
|
|
{
|
|
randomize(true);
|
|
index = 0;
|
|
}
|
|
|
|
alGenBuffers(bIDs.length, bIDs.ptr);
|
|
|
|
// If something fails, clean everything up.
|
|
scope(failure)
|
|
{
|
|
// This block is only executed if an exception is thrown.
|
|
|
|
if(fileHandle) cpp_closeAVFile(fileHandle);
|
|
|
|
fileHandle = null;
|
|
audioHandle = null;
|
|
|
|
alSourceStop(sID);
|
|
alDeleteSources(1, &sID);
|
|
alDeleteBuffers(bIDs.length, bIDs.ptr);
|
|
|
|
checkALError("cleaning up after music failure");
|
|
|
|
sID = 0;
|
|
bIDs[] = 0;
|
|
|
|
// Try the next track if playNext is called again
|
|
index++;
|
|
|
|
// The function exits here.
|
|
}
|
|
|
|
if(checkALError())
|
|
fail("Couldn't generate buffers");
|
|
|
|
fileHandle = cpp_openAVFile(toStringz(playlist[index]));
|
|
if(!fileHandle)
|
|
fail("Unable to open " ~ playlist[index]);
|
|
|
|
audioHandle = cpp_getAVAudioStream(fileHandle, 0);
|
|
if(!audioHandle)
|
|
fail("Unable to load music track " ~ playlist[index]);
|
|
|
|
int ch, bits, rate;
|
|
if(cpp_getAVAudioInfo(audioHandle, &rate, &ch, &bits) != 0)
|
|
fail("Unable to get info for music track " ~ playlist[index]);
|
|
|
|
bufRate = rate;
|
|
bufFormat = 0;
|
|
if(bits == 8)
|
|
{
|
|
if(ch == 1) bufFormat = AL_FORMAT_MONO8;
|
|
if(ch == 2) bufFormat = AL_FORMAT_STEREO8;
|
|
if(alIsExtensionPresent("AL_EXT_MCFORMATS"))
|
|
{
|
|
if(ch == 4) bufFormat = alGetEnumValue("AL_FORMAT_QUAD8");
|
|
if(ch == 6) bufFormat = alGetEnumValue("AL_FORMAT_51CHN8");
|
|
}
|
|
}
|
|
if(bits == 16)
|
|
{
|
|
if(ch == 1) bufFormat = AL_FORMAT_MONO16;
|
|
if(ch == 2) bufFormat = AL_FORMAT_STEREO16;
|
|
if(alIsExtensionPresent("AL_EXT_MCFORMATS"))
|
|
{
|
|
if(ch == 4) bufFormat = alGetEnumValue("AL_FORMAT_QUAD16");
|
|
if(ch == 6) bufFormat = alGetEnumValue("AL_FORMAT_51CHN16");
|
|
}
|
|
}
|
|
|
|
if(bufFormat == 0)
|
|
fail(format("Unhandled format (%d channels, %d bits) for music track %s", ch, bits, playlist[index]));
|
|
|
|
foreach(int i, ref b; bIDs)
|
|
{
|
|
int length = cpp_getAVAudioData(audioHandle, outData.ptr, outData.length);
|
|
if(length) alBufferData(b, bufFormat, outData.ptr, length, bufRate);
|
|
if(length == 0 || checkALError())
|
|
{
|
|
if(i == 0)
|
|
fail("No audio data in music track " ~ playlist[index]);
|
|
|
|
alDeleteBuffers(bIDs.length-i, bIDs.ptr+i);
|
|
checkALError("running alDeleteBuffers");
|
|
bIDs[i..$] = 0;
|
|
break;
|
|
}
|
|
}
|
|
|
|
alSourceQueueBuffers(sID, bIDs.length, bIDs.ptr);
|
|
alSourcePlay(sID);
|
|
if(checkALError())
|
|
fail("Unable to start music track " ~ playlist[index]);
|
|
|
|
index++;
|
|
return;
|
|
}
|
|
|
|
// Start playing the jukebox
|
|
void enableMusic()
|
|
{
|
|
if(!config.useMusic) return;
|
|
|
|
musicOn = true;
|
|
fading = Fade.None;
|
|
playNext();
|
|
}
|
|
|
|
// Disable music
|
|
void disableMusic()
|
|
{
|
|
if(fileHandle) cpp_closeAVFile(fileHandle);
|
|
fileHandle = null;
|
|
audioHandle = null;
|
|
|
|
if(sID)
|
|
{
|
|
alSourceStop(sID);
|
|
alDeleteSources(1, &sID);
|
|
checkALError("disabling music");
|
|
sID = 0;
|
|
}
|
|
|
|
alDeleteBuffers(bIDs.length, bIDs.ptr);
|
|
checkALError("deleting music buffers");
|
|
bIDs[] = 0;
|
|
|
|
musicOn = false;
|
|
}
|
|
|
|
// Pause current track
|
|
void pauseMusic()
|
|
{
|
|
fading = Fade.Out;
|
|
}
|
|
|
|
// Resume. Can also be called in place of enableMusic for fading in.
|
|
void resumeMusic()
|
|
{
|
|
if(!config.useMusic) return;
|
|
|
|
volume = 0.0;
|
|
fading = Fade.In;
|
|
musicOn = true;
|
|
if(sID) addTime(0);
|
|
else playNext();
|
|
}
|
|
|
|
// Checks if a stream is playing, filling more data as needed, and restarting
|
|
// if it stalled or was paused.
|
|
private bool isPlaying()
|
|
{
|
|
if(!sID) return false;
|
|
|
|
ALint count;
|
|
alGetSourcei(sID, AL_BUFFERS_PROCESSED, &count);
|
|
if(checkALError("in isPlaying()")) return false;
|
|
|
|
for(int i = 0;i < count;i++)
|
|
{
|
|
int length = cpp_getAVAudioData(audioHandle, outData.ptr, outData.length);
|
|
if(length <= 0)
|
|
{
|
|
if(i == 0)
|
|
{
|
|
ALint state;
|
|
alGetSourcei(sID, AL_SOURCE_STATE, &state);
|
|
if(checkALError() || state == AL_STOPPED)
|
|
return false;
|
|
}
|
|
break;
|
|
}
|
|
|
|
ALuint bid;
|
|
alSourceUnqueueBuffers(sID, 1, &bid);
|
|
if(checkALError() == AL_NO_ERROR)
|
|
{
|
|
alBufferData(bid, bufFormat, outData.ptr, length, bufRate);
|
|
alSourceQueueBuffers(sID, 1, &bid);
|
|
checkALError();
|
|
}
|
|
}
|
|
|
|
ALint state = AL_PLAYING;
|
|
alGetSourcei(sID, AL_SOURCE_STATE, &state);
|
|
if(state != AL_PLAYING) alSourcePlay(sID);
|
|
return (checkALError() == AL_NO_ERROR);
|
|
}
|
|
|
|
// Check if the music has died. This function is also used for fading.
|
|
void addTime(float time)
|
|
{
|
|
if(!musicOn) return;
|
|
|
|
if(!isPlaying()) playNext();
|
|
|
|
if(fading)
|
|
{
|
|
// Fade the volume
|
|
if(fading == Fade.In)
|
|
{
|
|
volume += fadeInRate * time;
|
|
if(volume >= maxVolume)
|
|
{
|
|
fading = Fade.None;
|
|
volume = maxVolume;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
assert(fading == Fade.Out);
|
|
volume -= fadeOutRate * time;
|
|
if(volume <= 0.0)
|
|
{
|
|
fading = Fade.None;
|
|
volume = 0.0;
|
|
|
|
// We are done fading out, disable music. Don't call
|
|
// enableMusic (or isPlaying) unless you want it to start
|
|
// again.
|
|
if(sID) alSourcePause(sID);
|
|
musicOn = false;
|
|
}
|
|
}
|
|
|
|
// Set the new volume
|
|
if(sID) alSourcef(sID, AL_GAIN, volume);
|
|
}
|
|
}
|
|
}
|