1 | /**************************************************************************/ |
2 | /* audio_stream_player.cpp */ |
3 | /**************************************************************************/ |
4 | /* This file is part of: */ |
5 | /* GODOT ENGINE */ |
6 | /* https://godotengine.org */ |
7 | /**************************************************************************/ |
8 | /* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */ |
9 | /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */ |
10 | /* */ |
11 | /* Permission is hereby granted, free of charge, to any person obtaining */ |
12 | /* a copy of this software and associated documentation files (the */ |
13 | /* "Software"), to deal in the Software without restriction, including */ |
14 | /* without limitation the rights to use, copy, modify, merge, publish, */ |
15 | /* distribute, sublicense, and/or sell copies of the Software, and to */ |
16 | /* permit persons to whom the Software is furnished to do so, subject to */ |
17 | /* the following conditions: */ |
18 | /* */ |
19 | /* The above copyright notice and this permission notice shall be */ |
20 | /* included in all copies or substantial portions of the Software. */ |
21 | /* */ |
22 | /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */ |
23 | /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */ |
24 | /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */ |
25 | /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */ |
26 | /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */ |
27 | /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */ |
28 | /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ |
29 | /**************************************************************************/ |
30 | |
31 | #include "audio_stream_player.h" |
32 | |
33 | #include "core/config/engine.h" |
34 | #include "core/math/audio_frame.h" |
35 | #include "servers/audio_server.h" |
36 | |
37 | void AudioStreamPlayer::_notification(int p_what) { |
38 | switch (p_what) { |
39 | case NOTIFICATION_ENTER_TREE: { |
40 | if (autoplay && !Engine::get_singleton()->is_editor_hint()) { |
41 | play(); |
42 | } |
43 | set_stream_paused(false); |
44 | } break; |
45 | |
46 | case NOTIFICATION_INTERNAL_PROCESS: { |
47 | Vector<Ref<AudioStreamPlayback>> playbacks_to_remove; |
48 | for (Ref<AudioStreamPlayback> &playback : stream_playbacks) { |
49 | if (playback.is_valid() && !AudioServer::get_singleton()->is_playback_active(playback) && !AudioServer::get_singleton()->is_playback_paused(playback)) { |
50 | playbacks_to_remove.push_back(playback); |
51 | } |
52 | } |
53 | // Now go through and remove playbacks that have finished. Removing elements from a Vector in a range based for is asking for trouble. |
54 | for (Ref<AudioStreamPlayback> &playback : playbacks_to_remove) { |
55 | stream_playbacks.erase(playback); |
56 | } |
57 | if (!playbacks_to_remove.is_empty() && stream_playbacks.is_empty()) { |
58 | // This node is no longer actively playing audio. |
59 | active.clear(); |
60 | set_process_internal(false); |
61 | } |
62 | if (!playbacks_to_remove.is_empty()) { |
63 | emit_signal(SNAME("finished" )); |
64 | } |
65 | } break; |
66 | |
67 | case NOTIFICATION_EXIT_TREE: { |
68 | set_stream_paused(true); |
69 | } break; |
70 | |
71 | case NOTIFICATION_PREDELETE: { |
72 | for (Ref<AudioStreamPlayback> &playback : stream_playbacks) { |
73 | AudioServer::get_singleton()->stop_playback_stream(playback); |
74 | } |
75 | stream_playbacks.clear(); |
76 | } break; |
77 | |
78 | case NOTIFICATION_PAUSED: { |
79 | if (!can_process()) { |
80 | // Node can't process so we start fading out to silence |
81 | set_stream_paused(true); |
82 | } |
83 | } break; |
84 | |
85 | case NOTIFICATION_UNPAUSED: { |
86 | set_stream_paused(false); |
87 | } break; |
88 | } |
89 | } |
90 | |
91 | void AudioStreamPlayer::set_stream(Ref<AudioStream> p_stream) { |
92 | stop(); |
93 | stream = p_stream; |
94 | } |
95 | |
96 | Ref<AudioStream> AudioStreamPlayer::get_stream() const { |
97 | return stream; |
98 | } |
99 | |
100 | void AudioStreamPlayer::set_volume_db(float p_volume) { |
101 | volume_db = p_volume; |
102 | |
103 | Vector<AudioFrame> volume_vector = _get_volume_vector(); |
104 | for (Ref<AudioStreamPlayback> &playback : stream_playbacks) { |
105 | AudioServer::get_singleton()->set_playback_all_bus_volumes_linear(playback, volume_vector); |
106 | } |
107 | } |
108 | |
109 | float AudioStreamPlayer::get_volume_db() const { |
110 | return volume_db; |
111 | } |
112 | |
113 | void AudioStreamPlayer::set_pitch_scale(float p_pitch_scale) { |
114 | ERR_FAIL_COND(!(p_pitch_scale > 0.0)); |
115 | pitch_scale = p_pitch_scale; |
116 | |
117 | for (Ref<AudioStreamPlayback> &playback : stream_playbacks) { |
118 | AudioServer::get_singleton()->set_playback_pitch_scale(playback, pitch_scale); |
119 | } |
120 | } |
121 | |
122 | float AudioStreamPlayer::get_pitch_scale() const { |
123 | return pitch_scale; |
124 | } |
125 | |
126 | void AudioStreamPlayer::set_max_polyphony(int p_max_polyphony) { |
127 | if (p_max_polyphony > 0) { |
128 | max_polyphony = p_max_polyphony; |
129 | } |
130 | } |
131 | |
132 | int AudioStreamPlayer::get_max_polyphony() const { |
133 | return max_polyphony; |
134 | } |
135 | |
136 | void AudioStreamPlayer::play(float p_from_pos) { |
137 | if (stream.is_null()) { |
138 | return; |
139 | } |
140 | ERR_FAIL_COND_MSG(!is_inside_tree(), "Playback can only happen when a node is inside the scene tree" ); |
141 | if (stream->is_monophonic() && is_playing()) { |
142 | stop(); |
143 | } |
144 | Ref<AudioStreamPlayback> stream_playback = stream->instantiate_playback(); |
145 | ERR_FAIL_COND_MSG(stream_playback.is_null(), "Failed to instantiate playback." ); |
146 | |
147 | AudioServer::get_singleton()->start_playback_stream(stream_playback, bus, _get_volume_vector(), p_from_pos, pitch_scale); |
148 | stream_playbacks.push_back(stream_playback); |
149 | active.set(); |
150 | set_process_internal(true); |
151 | while (stream_playbacks.size() > max_polyphony) { |
152 | AudioServer::get_singleton()->stop_playback_stream(stream_playbacks[0]); |
153 | stream_playbacks.remove_at(0); |
154 | } |
155 | } |
156 | |
157 | void AudioStreamPlayer::seek(float p_seconds) { |
158 | if (is_playing()) { |
159 | stop(); |
160 | play(p_seconds); |
161 | } |
162 | } |
163 | |
164 | void AudioStreamPlayer::stop() { |
165 | for (Ref<AudioStreamPlayback> &playback : stream_playbacks) { |
166 | AudioServer::get_singleton()->stop_playback_stream(playback); |
167 | } |
168 | stream_playbacks.clear(); |
169 | active.clear(); |
170 | set_process_internal(false); |
171 | } |
172 | |
173 | bool AudioStreamPlayer::is_playing() const { |
174 | for (const Ref<AudioStreamPlayback> &playback : stream_playbacks) { |
175 | if (AudioServer::get_singleton()->is_playback_active(playback)) { |
176 | return true; |
177 | } |
178 | } |
179 | return false; |
180 | } |
181 | |
182 | float AudioStreamPlayer::get_playback_position() { |
183 | // Return the playback position of the most recently started playback stream. |
184 | if (!stream_playbacks.is_empty()) { |
185 | return AudioServer::get_singleton()->get_playback_position(stream_playbacks[stream_playbacks.size() - 1]); |
186 | } |
187 | return 0; |
188 | } |
189 | |
190 | void AudioStreamPlayer::set_bus(const StringName &p_bus) { |
191 | bus = p_bus; |
192 | for (const Ref<AudioStreamPlayback> &playback : stream_playbacks) { |
193 | AudioServer::get_singleton()->set_playback_bus_exclusive(playback, p_bus, _get_volume_vector()); |
194 | } |
195 | } |
196 | |
197 | StringName AudioStreamPlayer::get_bus() const { |
198 | for (int i = 0; i < AudioServer::get_singleton()->get_bus_count(); i++) { |
199 | if (AudioServer::get_singleton()->get_bus_name(i) == String(bus)) { |
200 | return bus; |
201 | } |
202 | } |
203 | return SceneStringNames::get_singleton()->Master; |
204 | } |
205 | |
206 | void AudioStreamPlayer::set_autoplay(bool p_enable) { |
207 | autoplay = p_enable; |
208 | } |
209 | |
210 | bool AudioStreamPlayer::is_autoplay_enabled() { |
211 | return autoplay; |
212 | } |
213 | |
214 | void AudioStreamPlayer::set_mix_target(MixTarget p_target) { |
215 | mix_target = p_target; |
216 | } |
217 | |
218 | AudioStreamPlayer::MixTarget AudioStreamPlayer::get_mix_target() const { |
219 | return mix_target; |
220 | } |
221 | |
222 | void AudioStreamPlayer::_set_playing(bool p_enable) { |
223 | if (p_enable) { |
224 | play(); |
225 | } else { |
226 | stop(); |
227 | } |
228 | } |
229 | |
230 | bool AudioStreamPlayer::_is_active() const { |
231 | for (const Ref<AudioStreamPlayback> &playback : stream_playbacks) { |
232 | if (AudioServer::get_singleton()->is_playback_active(playback)) { |
233 | return true; |
234 | } |
235 | } |
236 | return false; |
237 | } |
238 | |
239 | void AudioStreamPlayer::set_stream_paused(bool p_pause) { |
240 | // TODO this does not have perfect recall, fix that maybe? If there are zero playbacks registered with the AudioServer, this bool isn't persisted. |
241 | for (Ref<AudioStreamPlayback> &playback : stream_playbacks) { |
242 | AudioServer::get_singleton()->set_playback_paused(playback, p_pause); |
243 | } |
244 | } |
245 | |
246 | bool AudioStreamPlayer::get_stream_paused() const { |
247 | // There's currently no way to pause some playback streams but not others. Check the first and don't bother looking at the rest. |
248 | if (!stream_playbacks.is_empty()) { |
249 | return AudioServer::get_singleton()->is_playback_paused(stream_playbacks[0]); |
250 | } |
251 | return false; |
252 | } |
253 | |
254 | Vector<AudioFrame> AudioStreamPlayer::_get_volume_vector() { |
255 | Vector<AudioFrame> volume_vector; |
256 | // We need at most four stereo pairs (for 7.1 systems). |
257 | volume_vector.resize(4); |
258 | |
259 | // Initialize the volume vector to zero. |
260 | for (AudioFrame &channel_volume_db : volume_vector) { |
261 | channel_volume_db = AudioFrame(0, 0); |
262 | } |
263 | |
264 | float volume_linear = Math::db_to_linear(volume_db); |
265 | |
266 | // Set the volume vector up according to the speaker mode and mix target. |
267 | // TODO do we need to scale the volume down when we output to more channels? |
268 | if (AudioServer::get_singleton()->get_speaker_mode() == AudioServer::SPEAKER_MODE_STEREO) { |
269 | volume_vector.write[0] = AudioFrame(volume_linear, volume_linear); |
270 | } else { |
271 | switch (mix_target) { |
272 | case MIX_TARGET_STEREO: { |
273 | volume_vector.write[0] = AudioFrame(volume_linear, volume_linear); |
274 | } break; |
275 | case MIX_TARGET_SURROUND: { |
276 | // TODO Make sure this is right. |
277 | volume_vector.write[0] = AudioFrame(volume_linear, volume_linear); |
278 | volume_vector.write[1] = AudioFrame(volume_linear, /* LFE= */ 1.0f); |
279 | volume_vector.write[2] = AudioFrame(volume_linear, volume_linear); |
280 | volume_vector.write[3] = AudioFrame(volume_linear, volume_linear); |
281 | } break; |
282 | case MIX_TARGET_CENTER: { |
283 | // TODO Make sure this is right. |
284 | volume_vector.write[1] = AudioFrame(volume_linear, /* LFE= */ 1.0f); |
285 | } break; |
286 | } |
287 | } |
288 | return volume_vector; |
289 | } |
290 | |
291 | void AudioStreamPlayer::_validate_property(PropertyInfo &p_property) const { |
292 | if (p_property.name == "bus" ) { |
293 | String options; |
294 | for (int i = 0; i < AudioServer::get_singleton()->get_bus_count(); i++) { |
295 | if (i > 0) { |
296 | options += "," ; |
297 | } |
298 | String name = AudioServer::get_singleton()->get_bus_name(i); |
299 | options += name; |
300 | } |
301 | |
302 | p_property.hint_string = options; |
303 | } |
304 | } |
305 | |
306 | void AudioStreamPlayer::_bus_layout_changed() { |
307 | notify_property_list_changed(); |
308 | } |
309 | |
310 | bool AudioStreamPlayer::has_stream_playback() { |
311 | return !stream_playbacks.is_empty(); |
312 | } |
313 | |
314 | Ref<AudioStreamPlayback> AudioStreamPlayer::get_stream_playback() { |
315 | ERR_FAIL_COND_V_MSG(stream_playbacks.is_empty(), Ref<AudioStreamPlayback>(), "Player is inactive. Call play() before requesting get_stream_playback()." ); |
316 | return stream_playbacks[stream_playbacks.size() - 1]; |
317 | } |
318 | |
319 | void AudioStreamPlayer::_bind_methods() { |
320 | ClassDB::bind_method(D_METHOD("set_stream" , "stream" ), &AudioStreamPlayer::set_stream); |
321 | ClassDB::bind_method(D_METHOD("get_stream" ), &AudioStreamPlayer::get_stream); |
322 | |
323 | ClassDB::bind_method(D_METHOD("set_volume_db" , "volume_db" ), &AudioStreamPlayer::set_volume_db); |
324 | ClassDB::bind_method(D_METHOD("get_volume_db" ), &AudioStreamPlayer::get_volume_db); |
325 | |
326 | ClassDB::bind_method(D_METHOD("set_pitch_scale" , "pitch_scale" ), &AudioStreamPlayer::set_pitch_scale); |
327 | ClassDB::bind_method(D_METHOD("get_pitch_scale" ), &AudioStreamPlayer::get_pitch_scale); |
328 | |
329 | ClassDB::bind_method(D_METHOD("play" , "from_position" ), &AudioStreamPlayer::play, DEFVAL(0.0)); |
330 | ClassDB::bind_method(D_METHOD("seek" , "to_position" ), &AudioStreamPlayer::seek); |
331 | ClassDB::bind_method(D_METHOD("stop" ), &AudioStreamPlayer::stop); |
332 | |
333 | ClassDB::bind_method(D_METHOD("is_playing" ), &AudioStreamPlayer::is_playing); |
334 | ClassDB::bind_method(D_METHOD("get_playback_position" ), &AudioStreamPlayer::get_playback_position); |
335 | |
336 | ClassDB::bind_method(D_METHOD("set_bus" , "bus" ), &AudioStreamPlayer::set_bus); |
337 | ClassDB::bind_method(D_METHOD("get_bus" ), &AudioStreamPlayer::get_bus); |
338 | |
339 | ClassDB::bind_method(D_METHOD("set_autoplay" , "enable" ), &AudioStreamPlayer::set_autoplay); |
340 | ClassDB::bind_method(D_METHOD("is_autoplay_enabled" ), &AudioStreamPlayer::is_autoplay_enabled); |
341 | |
342 | ClassDB::bind_method(D_METHOD("set_mix_target" , "mix_target" ), &AudioStreamPlayer::set_mix_target); |
343 | ClassDB::bind_method(D_METHOD("get_mix_target" ), &AudioStreamPlayer::get_mix_target); |
344 | |
345 | ClassDB::bind_method(D_METHOD("_set_playing" , "enable" ), &AudioStreamPlayer::_set_playing); |
346 | ClassDB::bind_method(D_METHOD("_is_active" ), &AudioStreamPlayer::_is_active); |
347 | |
348 | ClassDB::bind_method(D_METHOD("set_stream_paused" , "pause" ), &AudioStreamPlayer::set_stream_paused); |
349 | ClassDB::bind_method(D_METHOD("get_stream_paused" ), &AudioStreamPlayer::get_stream_paused); |
350 | |
351 | ClassDB::bind_method(D_METHOD("set_max_polyphony" , "max_polyphony" ), &AudioStreamPlayer::set_max_polyphony); |
352 | ClassDB::bind_method(D_METHOD("get_max_polyphony" ), &AudioStreamPlayer::get_max_polyphony); |
353 | |
354 | ClassDB::bind_method(D_METHOD("has_stream_playback" ), &AudioStreamPlayer::has_stream_playback); |
355 | ClassDB::bind_method(D_METHOD("get_stream_playback" ), &AudioStreamPlayer::get_stream_playback); |
356 | |
357 | ADD_PROPERTY(PropertyInfo(Variant::OBJECT, "stream" , PROPERTY_HINT_RESOURCE_TYPE, "AudioStream" ), "set_stream" , "get_stream" ); |
358 | ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "volume_db" , PROPERTY_HINT_RANGE, "-80,24,suffix:dB" ), "set_volume_db" , "get_volume_db" ); |
359 | ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "pitch_scale" , PROPERTY_HINT_RANGE, "0.01,4,0.01,or_greater" ), "set_pitch_scale" , "get_pitch_scale" ); |
360 | ADD_PROPERTY(PropertyInfo(Variant::BOOL, "playing" , PROPERTY_HINT_NONE, "" , PROPERTY_USAGE_EDITOR), "_set_playing" , "is_playing" ); |
361 | ADD_PROPERTY(PropertyInfo(Variant::BOOL, "autoplay" ), "set_autoplay" , "is_autoplay_enabled" ); |
362 | ADD_PROPERTY(PropertyInfo(Variant::BOOL, "stream_paused" , PROPERTY_HINT_NONE, "" ), "set_stream_paused" , "get_stream_paused" ); |
363 | ADD_PROPERTY(PropertyInfo(Variant::INT, "mix_target" , PROPERTY_HINT_ENUM, "Stereo,Surround,Center" ), "set_mix_target" , "get_mix_target" ); |
364 | ADD_PROPERTY(PropertyInfo(Variant::INT, "max_polyphony" , PROPERTY_HINT_NONE, "" ), "set_max_polyphony" , "get_max_polyphony" ); |
365 | ADD_PROPERTY(PropertyInfo(Variant::STRING_NAME, "bus" , PROPERTY_HINT_ENUM, "" ), "set_bus" , "get_bus" ); |
366 | |
367 | ADD_SIGNAL(MethodInfo("finished" )); |
368 | |
369 | BIND_ENUM_CONSTANT(MIX_TARGET_STEREO); |
370 | BIND_ENUM_CONSTANT(MIX_TARGET_SURROUND); |
371 | BIND_ENUM_CONSTANT(MIX_TARGET_CENTER); |
372 | } |
373 | |
374 | AudioStreamPlayer::AudioStreamPlayer() { |
375 | AudioServer::get_singleton()->connect("bus_layout_changed" , callable_mp(this, &AudioStreamPlayer::_bus_layout_changed)); |
376 | } |
377 | |
378 | AudioStreamPlayer::~AudioStreamPlayer() { |
379 | } |
380 | |