Sound Manager Plugin
My roles in the project
- Software designer
- Programmer
- Software engineer for the plugin and the demo
- UI and UX designer for the demo and partly for the plugin in-engine dock
What is the Sound Manager Plugin?
The Sound Manager is a module designed to give the user better control over the audio of their games. By using it, it is possible to play every sound of the game with simple method calls. No more long AudioStreamPlayer lists inside your scenes nor long methods to handle the audio inside every script.
This plugin has been made to facilitate the Sound Manager Module configuration process, by offering an UI in the editor to handle every part of it. Specifically, it was made to:
- Automatically set the Sound Manager module scene as an autoload inside the project
- Create a dock in the editor to make the configuration of the Sound Manager easier
Current version
Sound Manager Plugin version 4, running with the Sound Manager Module version 5.
Developed and tested with Godot Engine 4.3.
Configuration
- Copy & paste the folder
sound_manager
in theres://addons/
path from your project (this is the standard path for plugins in Godot). - To activate/deactivate the plugin, go to
Project/Project Settings/Plugins
, locate the plugin entry and change its status. - Once the plugin is activated, a dock called
SoundManager
will appear in the editor. Here you can change all the sounds settings, from default properties (volume, pitch) to AudioBus ("Master" is the fallback"). - Save the current scene (press the shortcut CTRL + S ).
Dock configuration
The UI consists of three sections to facilitate the configuration and use of the Sound Manager.
This UI:
- Allows you to change the default sound properties (volume and pitch) for each type of sound
- Allows you to set the audiobuses available in your project.
- Displays the sound files available in your resources.
- Allows you to easily build and edit the Audio FIles Dictionary.
"Jump" : "res://sfx_audio_jump.ogg"
.The dictionary is located inside the "SoundManager_config.gd" file, but you can just use the UI to edit it. You can place it wherever you want inside your project directory.
Another way to play the sounds is to use the absolute path that leads to the file, like
SoundManager.play_bgm("res://Audio/BGM/background_music.ogg")
.The dock also comes with an additional button:
Advanced Options
to access some more options. The ones currently available are:
Preload resources
: setting this true will make the SoundManager load every audio file located on the given paths at its_ready()
. Note that this may slow down game start, especially in projects with a long list of audio files, but will make playing sounds faster.Preinstantiate nodes
: you can tell the plugin to instantiate every node at startup. This will work in synergy with the "multiple sounds at once" feature for BGS, SE and ME. Note that this will make playing multiple sounds faster but may slow down games start and even the game itself, especially in bigger projects.
How does it work
Sound Types
The Sound Manager is able to manage 4 different sound types:
- BGM: background music, usually used to play and loop music on the background during scenes and gameplay
- BGS: background sound, usually used to play and loop sounds (like rain, or a crowd talking) on the background during scenes and gameplay
- SE: sound effects, usually used to play a short sound in certain moments during scenes or gameplay
- ME: music effects, usually used to play a short music in certain moments during scenes or gameplay (like a fanfare or a jingle) Note that there is no practical difference between this sound types. The only difference between them are the AudioBusses used and the default sound type properties. This means that you can use them however you like and however you think it fits your game and workflow.
Methods
The methods you'll probably use the most are just 11, plus some useful setters and getters.
All methods of the Sound Manager are accessed from the singleton called "SoundManager".
Sounds Handling
The play and fade_in methods exists in 4 forms: play/fade_in_bgm
, play/fade_in_bgs
, play/fade_in_sfx
, and play/fade_in_me
. They work in the same way.
func play(sound : String, from_position : float = 0.0, volume_db : float = -81, pitch_scale : float = -1, sound_to_override : String = "") -> void
:
this method lets you play the selected audio, passed as a string. If the audio is already playing, it will be restarted.
Theaudio
argument is the sound name. It can be an absolute path or the name you gave to the sound in the Audio Files Dictionary.
Thefrom_position
argument allows you to choose where the sound is going to start playing from. It's a float value that represents the track position in seconds. Default is 0.0.
volume_db
andpitch_scale
are respectively the values for the volume (in DB) and pitch for the sound track you want to play. If left to default the Sound Manager will just use the default settings you saved for that sound type from the dock.
sound_to_override
allows you to tell the module to not just play the sound, but to replace an already playing sound before doing that. If left blank, it will not do that and just play the new sound alongside the other.func stop(sound : String) -> void
:
this method lets you stop the specified stream from playing. The argument should be the same name you used for theplay
method.func find_sound(sound : String) -> int
:
this method will return the index number of the sound you're looking for on the internal arrays used by the module. It will return -1 if the sound was not found.func find_sound(sound : String) -> int
:
this method will return the index number of the sound you're looking for on the internal arrays used by the module. It will return -1 if the sound was not found.func is_playing(sound : String) -> bool
:
this method returnstrue
if the selected stream is plaiyng andfalse
if not.pause(sound : String) -> void
:
this method allows you to pause a specified stream. Note that a paused sound is not a stopped one, so the methodis_playing
will still returntrue
.unpause(sound : String) -> void
:
this method allows you to unpause a specified stream.set_paused(sound : String, paused : bool) -> void
:
this method allows you to set if the stream is paused or not. Note that there's no difference betweenset_paused(audio, true)
andpause(audio)
, you can use the one you prefer.is_paused(sound : String) -> bool
:
this method returnstrue
if the specified sound is paused.
Getters and setters
func get_playing_sounds() -> Array
:
this method returns an array containing the names of the currently playing soundstreams.
func set_volume_db(volume_db : float) -> void
:
this method allows you to change the default value for the given sound type.volume_db
is the volume in decibels. (set_bgm_volume_db for bgm
)func get_volume_db() -> float
:
this method return the default volume for the given sound type. (get_bgm_volume_db
for bgm)func set_pitch_scale(pitch : float) -> void
:
this method allows you to set the default pitch scale for the given sound type. (set_bgm_pitch_scale
for bgm)func get_pitch_scale() -> float
:
this method returns the default pitch scale of the given sound type. (get_bgm_pitch_scale
for bgm)
func set_volume_db(volume_db : float, sound : String) -> void
:
this method allows you to change the volume of the specified sound.func get_volume_db(sound : String) -> float
:
this method returns the volume of the specified sound. Returns -81.0 if the sound was not found.func set_pitch_scale(pitch_scale : float, sound : String) -> void
:
this method allows you to change the pitch scale of the specified sound.func get_pitch_scale(sound : String) -> float
:
this method returns the pitch scale of the specified sound. Returns -1.0 if the sound was not found.func get_default_sound_properties(sound_type : String) -> Dictionary
:
this method returns the default properties of the specified sound type. The argument must be a string like BGM, BGS, SE, or ME.
The return value will be a dictionary in the form of{Volume : value, Pitch : value}
.func set_sound_property(sound_type : String, property : String, value : float) -> void
:
this method allows you to change the given property for the given sound type. Sound type must be one of BGM, BGS, SE or ME. Property must be either Volume or Pitch.func get_audio_files_dictionary() -> Dictionary
:
this method returns the Audio Files Dictionary.func get_config_value(stream_name : String) -> String
:
this method returns the file name of the given stream name. Returnsnull
if an error occured.func set_config_key(new_stream_name : String, new_stream_file : String) -> void
:
this method allows the user to edit an existng value on the configuration dictionary, or add a new one in runtime.new_stream_name
is the name of your choice for the stream (the key in the dictionary), whilenew_stream_file
is the name of the file linked to it (the value in the dictionary).func add_to_dictionary(audio_name : String, audio_file : String) -> void
:
this method allows you to add a new voice to the dictionary in real time.audio_name
is the name which you are going to call the audio with (the key in the Dictionary).audio_file
is the file you want to play. It can be the file name (if the file is in the default audio dir path), or the absolute path for the file.func is_preload_resources_enabled() -> bool
:
this method returns true if the module has been set to preload resources. This method will also return false if you preload specific files from a list, as that doesn't ovveride thePreload Resources
variable.func enable_resource_preloading(enabled : bool = true) -> void
: this method allows you to enable resource preloading from script.
Resource preloading
There are also some useful methods to manage resource preloading:
func preload_resources_from_list(files_list : Array) -> void
:
this method allows you to preload only a specific list of audio files. The content of thefiles_list
array must be a recognizable sound name String, such as an absolute path, a sound name stored on theAudio_Files_Dictionary
or even an already loaded sound Resource. This method is especially useful when you want to preload only certain sounds and not all of them, maybe because you know you will need them on the specific scene you're programming. Note that if thePreload Resources
variable is enabled, this method will do nothing.func preload_resources_from_path(path : String) -> void
:
this method lets you preload every audio file located in a specific directory (passed via thepath
string argument). This is especially useful if you are using a different folder from the standard directory that you set on the dock for some audio files and want to preload them too without having to write a full list of files. This can be used alongside the automaticpreload process
.func preload_resource(file : Resource) -> void
:
this is mainly an internal method, but in any case you can still use it to preload a specific file. Thefile
argument must be an already loaded soundResource
. You can basically see this method as a way to store a loaded resource to use it on the Sound Manager as you please. The module will store this variable linking it to the file name as it would do with any other preloaded resource, so to play the sound you just have to use the file name or the sound name you used on the Audio Files Dictionary.func preload_resource_from_string(file : String) -> void
:
this is mainly an internal method, but in any case you can still use it to preload a specificfile
. For the file string argument rules, read the above rules aboutfiles_list
array rules. Although this method exists and can be used, it's probably better to use thepreload_resources_from_list
method for almost any uses of this feature.func unload_all_resources(force_unload : bool = false) -> void
:
this method allows you to unload every previously preloaded audio file. It's especially useful when used in combo with thepreload_resources_from_list
to unload at the end of a scene any resource you loaded at the start of the scene. Theforce_unload
argument (default:false
) will let you unload preloaded resources even if thePreload Resources
variable is set to on. Note that this will unload all preloaded resources, so it basically overwrites thePreload Resources
feature. If theforce_unload
argument is set to off, however, the method will do nothing if called whilePreload Resources
is on.func unload_resources_from_list(files_list : Array) -> void
:
this method allows you to pass a list of preloaded resources you want to unload. The files have to be Strings, but can be passed in any format (absolute path, file name, sound name). Note that thay can't be loaded Resources (why are you passing a loaded Resource if you want to unload it in the first place?)func unload_resource_from_string(file : String) -> void
:
this method allows you to unload a previously loaded resource passed by a string. The string can be passed in any format (absolute path, sound name, file name).func unload_resources_from_dir(path : String) -> void
:
this method lets you unload every audio file located in a specific directory (passed via thepath
string argument). This is especially useful if you preloaded a different folder from the standard directory that you set on the dock for some audio files and want to unload them too without having to write a full list of files. This can be used alongside the automatic preload process.
Node preinstantiation
There are also some useful methods to manage node preinstantiation:
func preinstantiate_nodes_from_path(path : String, sound_type : String = "") -> void
:
this method lets you preinstantiate every needed node on a specified directory (given its path). This can be used alongside the automatic preinstantiation process. You can also specify thesound_type
(BGM, BGS, ME, SE) to allow the Sound Manager to automatically set the correct bus. If you don't, it will temporary use the "Master" bus until you play the sound for the first time. Then, it will update it.func preinstantiate_nodes_from_list(files_list : Array, type_list : Array, all_same_type : bool = false) -> void
:
this method allows you to pass a list of files you want to preinstantiate a node for. The files have to be Strings, but can be passed in any format (absolute path or sound name). Note: thetype_list
argument is used to tell the module which type does any sound you passed have. The indexes of thetype_list
must coincide with the indexes of thefiles_list
. If theall_same_type
argument is passed as true, you can pass a single element array for thetype_list
implying that all the files you're passing are of the same sound type.func preinstantiate_node_from_string(file : String, sound_type : String = "") -> void
:
this method allows you to instantiate a node for a specific audio file passed by a string. The string can be passed in any format (absolute path or file name). Thesound_type
string is the type of the audio and must be either BGM, BGS, SE or ME. If left blank, the Sound Manager will use the "Master" bus for the node until you play it for the first time. Then, it will update it.func preinstantiate_node(stream : Resource, sound_type : String = "") -> void
:
this method allows you to instantiate a node for a specific audio file passed as an already loaded Resource. The file can be accessed afterwards with a sound name or a file name. Thesound_type
string is the type of the audio and must be either BGM, BGS, SE or ME. If left blank, the Sound Manager will use the "Master" bus for the node until you play it for the first time. Then, it will update it.func uninstantiate_all_nodes(force_uninstantiation : bool = false) -> void
:
this method allows you to uninstantiate every previously instantiated node. It's especially useful when used in combo with thepreinstantiate_nodes_from_list
to uninstantiate at the end of a scene any node you instantiated at the start of the scene. Theforce_uninstantiation
argument (default:false
) will let you uninstantiate nodes even if thePreinstantiate Nodes
variable is set to on. Note that this will uninstantiate all instantiated resources, so it basically overwrites thePreinstantiate Nodes
feature. If theforce_uninstantiation
argument is set to off, however, the method will do nothing if called whilePreinstantiate Nodes
is on.func uninstantiate_nodes_from_list(files_list : Array) -> void
:
this method allows you to pass a list of preinstantiated nodes you want to uninstantiate. The nodes have to be Strings, but can be passed in any format (absolute path or sound name).func uninstantiate_node_from_string(file : String) -> void
:
this method allows you to uninstantiate a previously instantiated node passed by a string. The string can be passed in any format (absolute path or file name).func uninstantiate_nodes_from_dir(path : String) -> void
:
this method lets you uninstantiate every instantiated node based on a directory content (given its path). This is especially useful if you are using a different folder from the standard directory that you set on the dock for some audio files and want to uninstantiate them too without having to write a full list of files. This can be used alongside the automatic preinstantiation process.func enable_nodes_preinstantiation(enabled : bool = true) -> void
:
this methods allows you to set the value on thePreinstantiate Nodes
variable.func is_preinstantiate_nodes_enabled() -> bool
:
this method returns true if the module has been set to preinstantiate nodes. This method will also return false if you instantiate specific nodes from a list, as that doesn't ovveride thePreinstantiate Nodes
variable.
Want more?
If the docs are not enough for you and you want a demo of configuration and usage, you can check out the official Sound Manager Demo which lets you give a sneak peek to the implementation and usage of this plugin.
Where can you find it?
You can find the source code for this plugin by clicking on the GitLab logo above or by clicking on this link.
You can also download the plugin on Itch by clicking on the button below or by going at this link. You will also find a downloadable zip file with the Demo project.
You can also find the plugin on the official Godot Asset Library by clicking on the link or by checking the AssetLib
tab on your engine editor and searching for the Sound Manager.
Comments and feedback
If you have issues, comments, questions about this code, you can contact me by writing a comment on the itch.io page opening an issue ticket on the GitLab page, via Twitter (find my profile link on the blog footer, and follow me while you're there!) or by sending me an email (you can find my contact info on the footer on this page).
Copyright
Sound Manager Module
Copyright © 2019-2024 Celeste Privitera
This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0. If a copy of the MPL was not distributed with this file, you can obtain one at https://mit-license.org//.
Sound Manager Plugin
Copyright © 2019-2024 Simón Olivo & Celeste Privitera
This Source Code Form is subject to the terms of the MIT License. If a copy of the license was not distributed with this file, you can obtain one at https://mit-license.org/.
Sound Manager Demo
Copyright © 2020-2024 Celeste Privitera
This Source Code Form is subject to the terms of the Mozilla Public License, v. 2.0. If a copy of the MPL was not distributed with this file, you can obtain one at https://mozilla.org/MPL/2.0/.