Aller au contenu

Playing sounds and music

Sound or music?

SFML provides two classes for playing audio: sf::Sound and sf::Music. They both provide more or less the same features, the main difference is how they work.

sf::Sound is a lightweight object that plays loaded audio data from an sf::SoundBuffer. It should be used for small sounds that can fit in memory and should suffer no lag when they are played. Examples are gun shots, foot steps, etc.

sf::Music doesn't load all the audio data into memory, instead it streams it on the fly from the source file. It is typically used to play compressed music that lasts several minutes, and would otherwise take many seconds to load and eat hundreds of MB in memory.

Loading and playing a sound

As mentioned above, the sound data is not stored directly in sf::Sound but in a separate class named sf::SoundBuffer. This class encapsulates the audio data, which is basically an array of 16-bit signed integers called "audio samples". A sample is the amplitude of the sound signal at a given point in time, and an array of samples therefore represents a full sound.

In fact, the sf::Sound/sf::SoundBuffer classes work the same way as sf::Sprite/sf::Texture from the graphics module. So if you understand how sprites and textures work together, you can apply the same concept to sounds and sound buffers.

You can load a sound buffer from a file on disk with its loadFromFile function or its corresponding constructor:

#include <SFML/Audio.hpp>

int main()
{
    sf::SoundBuffer buffer("sound.wav"); // Throws sf::Exception if an error occurs

    // OR

    sf::SoundBuffer buffer;
    if (!buffer.loadFromFile("sound.wav"))
        return -1;

    ...
}

As with everything else, you can also load an audio file from memory (loadFromMemory) or from a custom input stream (loadFromStream) along with corresponding constructors that throw if an error occurs.

SFML supports the audio file formats WAV, OGG/Vorbis and FLAC. MP3 is supported for decoding but not encoding.

You can also load a sound buffer directly from an array of samples, in the case they originate from another source:

std::vector<std::int16_t> samples = ...;

sf::SoundBuffer buffer(samples.data(), samples.size(), 2, 44100);

// OR

sf::SoundBuffer buffer;
if (!buffer.loadFromSamples(samples.data(), samples.size(), 2, 44100))
    // Handle error

Since loadFromSamples and its corresponding constructor loads a raw array of samples rather than an audio file, it requires additional arguments in order to have a complete description of the sound. The first one (third argument) is the number of channels; 1 channel defines a mono sound, 2 channels define a stereo sound, etc. The second additional attribute (fourth argument) is the sample rate; it defines how many samples must be played per second in order to reconstruct the original sound.

Now that the audio data is loaded, we can play it with an sf::Sound instance.

sf::SoundBuffer buffer;
// load something into the sound buffer...

sf::Sound sound(buffer);
sound.play();

The cool thing is that you can assign the same sound buffer to multiple sounds if you want. You can even play them together without any issues.

Sounds (and music) are played in a separate thread. This means that you are free to do whatever you want after calling play() (except destroying the sound or its data, of course), the sound will continue to play until it's finished or explicitly stopped.

Playing a music

Unlike sf::Sound, sf::Music doesn't pre-load the audio data. Instead it streams the data directly from the source. The initialization of music is thus more direct:

sf::Music music("music.ogg"); // Throws sf::Exception if an error occurs

// OR

sf::Music music;
if (!music.openFromFile("music.ogg"))
    return -1; // error

music.play();

It is important to note that, unlike most other SFML resources, the loading function is named openFromFile instead of loadFromFile. This is because the music is not really loaded. This function merely opens it. The data is only loaded later when the music is played. It also helps to keep in mind that the audio file has to remain available as long as it is played.

The other loading functions of sf::Music follow the same convention: openFromMemory, openFromStream.

What's next?

Now that you are able to load and play a sound or music, let's see what you can do with it.

To control playback, the following functions are available:

  • play starts or resumes playback
  • pause pauses playback
  • stop stops playback and rewind
  • setPlayingOffset changes the current playing position

Example:

// start playback
sound.play();

// advance to 2 seconds
sound.setPlayingOffset(sf::seconds(2.f));

// pause playback
sound.pause();

// resume playback
sound.play();

// stop playback and rewind
sound.stop();

The getStatus function returns the current status of a sound or music, you can use it to know whether it is stopped, playing or paused.

Sound and music playback is also controlled by a few attributes which can be changed at any moment.

The pitch is a factor that changes the perceived frequency of the sound: greater than 1 plays the sound at a higher pitch, less than 1 plays the sound at a lower pitch, and 1 leaves it unchanged. Changing the pitch has a side effect: it impacts the playing speed.

sound.setPitch(1.2f);

The volume is... the volume. The value ranges from 0 (mute) to 100 (full volume). The default value is 100, which means that you can't make a sound louder than its initial volume.

sound.setVolume(50.f);

The loop attribute controls whether the sound/music automatically loops or not. If it loops, it will restart playing from the beginning when it's finished, again and again until you explicitly call stop. If not set to loop, it will stop automatically when it's finished.

sound.setLoop(true);

More attributes are available, but they are related to spatialization and are explained in the corresponding tutorial.

Looping music

sf::Music provides the ability to loop a given segment of the audio. sf::Music::setLoopPoints sets the beginning and duration of the sound's looping sequence. Here's what that looks like:

sf::Music music("music.wav");
music.setLoopPoints({sf::milliseconds(500), sf::seconds(4)});

This code will loop a 4 second long segment of the music starting 500 milliseconds from the beginning. sf::Music may tweak this loop range to match up with the internally held samples so you can check the actual loop points with getLoopPoints.

Common mistakes

Destroyed sound buffer

The most common mistake is to let a sound buffer go out of scope (and therefore be destroyed) while a sound still uses it.

sf::Sound loadSound(const std::filesystem::path& filename)
{
    sf::SoundBuffer buffer(filename); // this buffer is local to the function, it will be destroyed...
    return sf::Sound(buffer);
} // ... here

sf::Sound sound = loadSound("s.wav");
sound.play(); // ERROR: the sound's buffer no longer exists, the behavior is undefined

Remember that a sound only keeps a pointer to the sound buffer that you give to it, it doesn't contain its own copy. You have to correctly manage the lifetime of your sound buffers so that they remain alive as long as they are used by sounds.

Destroying the music source while it plays

Remember that a music needs its source as long as it is played. A music file on your disk probably won't be deleted or moved while your application plays it, however things get more complicated when you play a music from a file in memory, or from a custom input stream:

// we start with a music file in memory (imagine that we extracted it from a zip archive)
std::vector<char> fileData = ...;

// we play it
sf::Music music;
music.openFromMemory(filedata.data(), fileData.size());
music.play();

// "ok, it seems that we don't need the source file any longer"
fileData.clear();

// ERROR: the music was still streaming the contents of fileData! The behavior is now undefined

sf::Music is not copyable

The final "mistake" is a reminder: the sf::Music class is not copyable, so you won't be allowed to do that:

sf::Music music;
sf::Music anotherMusic = music; // ERROR

void doSomething(sf::Music music)
{
    ...
}
sf::Music music;
doSomething(music); // ERROR (the function should take its argument by reference, not by value)

sf::Music is however moveable. You can transfer ownership of a music object like this:

class MyMusic
{
public:
    MyMusic(sf::Music&& music) : m_music(std::move(music))
    {
    }

    // ... other public functions

private:
    sf::Music m_music;
};

sf::Music music("sound.ogg");
MyMusic   myMusic(std::move(music));