This is my preferential player when I need to play various sound files in a project.
It is multi-platform and has no dependencies, other than what comes with Windows 10, the standard Linux kernel, MacOS 10.5 or later, and the Python Standard Library.
It contains enough functions to be useful, but not too many to be confusing, and I tried to keep the syntax and implementation super-simple and human readable.
When I built this module, I considered many factors that were important to me, like maintenance of code, ease of use, reliability and so forth. I used these methods below as they seem to be the best choices, considering the above factors.
In a nutshell:
-Linux will use the first available player in this order: gst-1.0-play, ffmpeg, gst playbin(built on the fly) or ALSA
-Linux will try to use gst-1.0-play first (usually present), if not present then
-Linux will try to use ffmpeg as its player (usually present), if not present then
-Linux will initialize a gstreamer playbin player (is supposed to always be present), if not present then
-Linux will play the sound with ALSA, and if not a .wav file will sound like white noise.
In Windows, the duration of the sound file is calculated and that duration is used to loop the sound file. In Linux, the duration of the sound file is calculated if it is a wav file, otherwise, a loop checks every 0.2 seconds to see if the sound file is playing, if not, it relaunches the sound. MacOS works similar to the Linux algorithm. So if you use .wav files instead of mp3s, you avoid a background loop that checks 5 times a second, and also the small gap in sound that can occur when the loop restarts (less than .2 seconds). See additional notes below about looping in a game loop.
To use the module simply add:
from preferredsoundplayer import *
and this will import all its functions.
The module essentially contains 3 functions for working with sound files:
yourSound = soundplay("yourfilename.mp3") # or just soundplay("yourfilename.mp3")
stopsound(yourSound)
getIsPlaying(yourSound)
And a few more for looping .wav files:
backgroundSong = loopwave("yourfilename.wav")
and
stoploop(backgroundSong)
Here are some examples on how to use them. Note that with 'soundplay' it can be used as a standalone function, but if you want to stop the file from playing, you will have to use the return value of soundplay. Read a little further and the examples hopefully will make sense.
soundplay("coolhipstersong.mp3") #-> this plays the mp3 file
mysong = soundplay("coolhipstersong.mp3") #-> this plays the mp3 and also returns a reference to the song.
stopsound(mysong) # -> this stops mysong, which you created in the line above
isitplaying = getIsPlaying(mysong) -> sets a variable to True or False, depending on if process is running
print(getIsPlaying(mysong)) -> prints True or False depending on if process is running
if getIsPlaying(mysong)==True:
print("Yes, your song is playing")
else:
print("Your song is not playing")
soundplay("coolhipsong.mp3",1) #-> this plays the mp3 file synchronously
or
soundplay("coolhipsong.mp3",block=True)
* Note: commands below will work, but you cannot stop the song, because your progam will be blocked until the song is done playing
mysong = soundplay("coolhipstersong.mp3",1) #-> this plays the wav file synchronously and also returns the song reference
or
mysong = soundplay("coolhipstersong.mp3",block=True) #-> this plays the wav file synchronously and also returns the song reference
myloop = loopwave("mybackgroundsong.wav")
This starts a background loop playing, but also returns a reference to the background process so it can be stopped.
stoploop(myloop)
Windows10 functions use the winmm.dll Windows Multimedia API calls using c function calls to play sounds.
See references:
“Programming Windows: the Definitive Guide to the WIN32 API, Chapter 22 Sound and Music Section III Advanced Topics ‘The MCI Command String Approach.’” Programming Windows: the Definitive Guide to the WIN32 API, by Charles Petzold, Microsoft Press, 1999.
https://stackoverflow.com/questions/22253074/how-to-play-or-open-mp3-or-wav-sound-file-in-c-program
https://github.com/michaelgundlach/mp3play
& https://github.com/TaylorSMarks/playsound/blob/master/playsound.py
This method of playing sounds allows for multiple simultaneous sounds, works well and has been used successfully in several projects. As long as this dynamically linked library is bundled with the current version of Windows, I plan to use this as the preferred method of playing sounds unless there is a compelling reason to change. In this case, I am using the reasoning "If it's not broke, don't fix it.". Another advantage is it plays mp3s and other formats as well, not just .wav files. It works well, is stable, loads and executes quickly, and has essentially never caused me any problems.
The Python winsound
module on the other hand, is at least to me a bit odd in its syntax, less intuitive, and only uses wave files. You basically can't play more than one wave at a time asynchronously. This is severely limiting, so I don't prefer it for playing sounds.
Calling the winsound.PlaySound module through the OS system works, but not does not execute as quickly. This may not be a bad approach, however, for background sounds whose fine-timing is not critical.
In this latest version, I use the winmm.dll mciSendString calls with additional specifications to loop, rather than using with winsound module, as it allows for multiple simultaneous loops if you would need but more importantly will allow looping of mp3 files, in addition to .wav files.
You can loop sounds by using OS system calls in the style of using command line instructions.
See https://pypi.org/project/oswaveplayer/ for an example.
This is not a bad approach, but there is a little delay with the sound launch using the command line version. This may not be a big issue for you when playing background music. Another way to play multiple background sounds at once would be to use another module or to add the oswaveplayer to your project with the import statement:
from oswaveplayer import oswaveplayer #(this can be installed with "pip install oswaveplayer")
then use:
backgroundSong = oswaveplayer.loopwave("yourfilename.wav")
and
oswaveplayer.stoploop(backgroundSong)
This is not a bad approach, but due to the perceptible delay in playing the sound, it is not preferred to me. You can also look over the source code to see how to launch sounds using this approach, as it is very basic.
As far as I know the gst-1.0-play command is usually available on linux distributions. It has almost always has been for me and its part of the gstreamer library. My understanding is it can be built from the gstreamer library and that library has been included with the standard Linux kernel for a long time. It may not be built and ready to go in a Linux disto though or even on path. So I first try this player. If that does not work, I try to play the file with ffmpeg command from the ffmpeg project. As far as I know, it is not part of the standard Linux kerrnel, but is usually present on most distros. If these 2 commands are not present, which is unusual, the program will use the gstreamer library, which should be present to make a playbin player using the gstreamer library. I do this a little differently than some other players, as I initialize a gst playbin player for each sound. (Otherwise, you will get quite a few warnings, some critical related to the internal looping of the gst player). If this fails for some reason (maybe in the future, some import statement changes, who knows??) it will play with the ALSA player. mp3s will not sound right, but like white noise.
afplay
system calls work great on 'MacOS'. I see no reason to invoke different methods at this point in time. I do not want to concern myself with trying to make this module work with significantly older versions of MacOS before afplay
was available. My perception is that people typically would rather use Linux on an old computer systems instead of loading old versions of MacOS. afplay
has been present for some time now on MacOS. Please contact me if you think this really needs to work on older versions of MacOS.
If you using a game loop in game building, you don't actually need to use these looping functions at all (although it may be a little more convenient). You may notice that this module, and other packages I have written, all contain a function called getIsPlaying(yoursound). You can simply implement a check in your game loop to see if yoursong is playing. If it is, don't do anything. If it is not, play the sound with yoursound = soundplay("yourfilename.wav")
. Maybe check every 10 frames or something like that in the game loop.
I have included these aliases, in case this module is dropped in as a replacement for my other 2 sound playing modules, oswaveplayer
and preferredwaveplayer
.
In other words, if someone has used those modules in a project and they need more functionality, they can import this one and it should work as it contains the same functions, but different implementation. These 3 function names are available, but they simply are references for backwards compatibility.
playwave = soundplay
stopwave = stopsound
loopwave = loopsound
Additionally, I included an alias/reference to the function named 'playsound', and if used, the default block will be true, or synchronous play. This way, the module can be used in place of the playsound module (https://github.com/TaylorSMarks/playsound/blob/master/playsound.py) with the same syntax. If the playsound module is no longer maintained or otherwise does not work for you, you can load this module and use the import statement.
Use:
from preferredwaveplayer import playsound
for backwards compatibility with the playsound module.
I do not worry about playing urls with this module though.
Update 8/2/2021 -Manual 'Garbage Collection' now being performed in Windows when the winmm.dll module is used.
I added garbage collection which I believe sorely needed for this module. Other players which I have cited have used the windows multimedia module to play sounds by issuing c type commands from python to the winmm.dll module. The problem is that memory is allocated for these sounds using the c language. The wimm.dll module sounds are supposed to be closed in order to reallocate that memory. In windows this can be done by using and event listenter to report when the playing of the sound is complete, and then it can be closed.
It turns out is easy to issue a play command from python using the c commands wrapper. However, it is much harder to set up an event listener and report it back to Python. If it can be done at all it would be hard very complicated to do.
So I created my own garbage collection algorithm from the Python side of things. Basically, every time a call to the winmm.dll module is called, that alias is added to a list. Also, all prior aliases in the list are checked to see if the sound is playing. If it is not, stop and close calls are issued to that sound alias, then the alias is removed from the garbage collection list.
In short before playing a new sound, prior sounds are checked to see if they have finished and then closed. There is no event listended for, but cleanup is done at every play of a sound.