Skip to main content
Android
iOS
macOS
Web
Windows
Electron
Flutter
React Native
React JS
Unity
Unreal Engine
Unreal (Blueprint)

3D Spatial audio

3D Spatial Audio brings the real-world sound experience to the virtual world, providing an immersive audio experience for users. Agora's spatial audio technology enables you to simulate the propagation characteristics of sound in a physical environment within a virtual interactive scene.

Spatial audio comparison

  • Ultra-realistic space shaping effect

    Utilize technologies such as range audio, sound blur, and air attenuation simulation to perfectly simulate the real auditory experience.

    • Set the spatial positions of users in real time to give a sense of change in the distance, direction, and orientation of other users.
    • Update the spatial position of the media player, to add a sense of space to background sounds, accompaniments, and other media resources.
    • Add 3D Spatial Audio effects such as sound blurring and air attenuation by adjusting audio settings to perfectly simulate the real audio experience.
  • 3D High Fidelity

    • Sound effects are processed and rendered based on the facial orientation, sound source orientation, and relative position of the sound source in 3D space.
    • Supports 48 kHz full-band sampling and 3D high-fidelity audio processing and rendering.
  • Multi-platform support

    Supports iOS, Android, macOS, Windows, Web, Unity, Flutter, React Native, Electron, Unreal and other platforms.

  • Ultra-low latency, low power consumption, and low cost

    The spatial audio algorithm adopts an advanced front-end processing mode and synchronizes spatial coordinates through cloud services. The end-cloud collaborative processing mode effectively reduces overall latency and power consumption.

Compared to traditional stereo, which relies on left and right channels, spatial audio technology greatly enhances the depth and realism of sound. The following table highlights these enhancements:

FeatureTraditional StereoAgora Spatial Audio
DimensionalityLeft and right dimensionsRepresents sound in a full 3D space using the x, y, and z axes of the world coordinate system, corresponding to right, top, and front dimensions
Spatial PerceptionAdjust the volume of the left and right channels to create a spatial soundUtilizes advanced spatial audio algorithms to create a realistic soundscape by manipulating parameters like distance, direction, and orientation
User ExperienceFlatImmersive, three-dimensional, and natural, delivering a realistic auditory experience

This page shows you how to implement 3D Spatial Audio in your app.

Understand the tech

Agora provides a local Cartesian coordinate system calculation scheme for setting up 3D Spatial Audio positions for users and the media player.

3D Spatial Audio for users

Use an instance of ILocalSpatialAudioEngine to implement 3D Spatial Audio. Call updateSelfPosition and updateRemotePosition to specify the spatial coordinates of local and remote users in the channel. Voice SDK calculates the relative positions of local and remote users. This enables local users to experience the 3D Spatial Audio of remote users.

SpatialAudio

3D Spatial Audio for media player

Call updateSelfPosition and updatePlayerPositionInfo methods of the ILocalSpatialAudioEngine class to update the spatial coordinates of local users and the media player. The SDK calculates the relative positions of local users and the media player. This enables local users to experience the 3D Spatial Audio effect of the media player.

SpatialAudio

Prerequisites

Ensure that you have implemented the SDK quickstart in your project.

Implement 3D Spatial Audio

The following figure shows the workflow you implement to provide 3D Spatial Audio for users and the media player:

Spatial audio steps

Initialize the spatial audio engine

Call the create method of the ILocalSpatialAudioEngine class to create an instance of the spatial audio engine. Then, call initialize to enable spatial audio.


_4
localSpatial = ILocalSpatialAudioEngine.create();
_4
LocalSpatialAudioConfig localSpatialAudioConfig = new LocalSpatialAudioConfig();
_4
localSpatialAudioConfig.mRtcEngine = engine;
_4
localSpatial.initialize(localSpatialAudioConfig);

Set the audio profile and scenario

To set the desired audio encoding properties, call setAudioProfile.

Call setAudioScenario to set the scenario to AUDIO_SCENARIO_GAME_STREAMING for the best sound quality effect.


_2
engine.setAudioProfile(io.agora.rtc2.Constants.AUDIO_PROFILE_MUSIC_STANDARD);
_2
engine.setAudioScenario(io.agora.rtc2.Constants.AUDIO_SCENARIO_GAME_STREAMING);

Set the audio reception range

To set the maximum number of audio streams that can be received within the audio receiving range, call setMaxAudioRecvCount. The recommended maxCount value is ≤ 16.

To set the maximum range of receivable audio, in meters, call setAudioRecvRange. The recommended value range is > 0.


_2
localSpatial.setMaxAudioRecvCount(2);
_2
localSpatial.setAudioRecvRange(AXIS_MAX_DISTANCE);

Update spatial position

In the user space audio scenario, call the updateSelfPosition and updateRemotePosition methods to update the positions of the local user and remote users respectively.

In the media player spatial audio scenario, call the updateSelfPosition and updatePlayerPositionInfo methods to update the position of the local user and the media player, respectively.

Typically, you call these methods when:

  • A new user joins the channel.
  • The relative positions of the local user, a remote user, or the media player changes.
  • There are other changes in your specific scenario.

_23
// Update local user position
_23
float[] pos = getVoicePosition(localIv);
_23
float[] forward = new float[]{1.0F, 0.0F, 0.0F};
_23
float[] right = new float[]{0.0F, 1.0F, 0.0F};
_23
float[] up = new float[]{0.0F, 0.0F, 1.0F};
_23
localSpatial.updateSelfPosition(pos, forward, right, up);
_23
_23
// Update the remote user's position
_23
if (remoteLeftTv.getTag() == null) {
_23
remoteLeftTv.setTag(uid);
_23
remoteLeftTv.setVisibility(View.VISIBLE);
_23
remoteLeftTv.setText(uid + "");
_23
RemoteVoicePositionInfo info = getVoicePositionInfo(remoteLeftTv);
_23
Log.d(TAG, "left remote user >> pos=" + Arrays.toString(info.position));
_23
localSpatial.updateRemotePosition(uid, info);
_23
remoteLeftTv.setOnClickListener(v -> showRemoteUserSettingDialog(uid));
_23
} else if (remoteRightTv.getTag() == null) {
_23
remoteRightTv.setTag(uid);
_23
remoteRightTv.setVisibility(View.VISIBLE);
_23
remoteRightTv.setText(uid + "");
_23
localSpatial.updateRemotePosition(uid, getVoicePositionInfo(remoteRightTv));
_23
remoteRightTv.setOnClickListener(v -> showRemoteUserSettingDialog(uid));
_23
}

Set spatial audio parameters

To set spatial audio parameters for the remote user or media player, call setRemoteUserSpatialAudioParams or setSpatialAudioParams. To implement specific sound effects, refer to the following parameter settings:

  • Air attenuation effect

    Set enable_air_absorb to true and set speaker_attenuation to the desired sound attenuation coefficient.

  • Sound blur effect

    Set enable_blur to true


_8
// Enable air absorption effect
_8
spatialAudioParams.enable_air_absorb = true;
_8
_8
// Enable sound blur effect
_8
spatialAudioParams.enable_blur = true;
_8
_8
// Set spatial audio parameters for the media player
_8
engine.setRemoteUserSpatialAudioParams(uid, spatialAudioParams);

Set up sound isolation (optional)

To define a sound isolation zone and set a sound attenuation coefficient call setZones. This feature simulates the real-world effect of sound attenuation when the sound source is inside a sound isolation zone while the receiver is outside or vice versa. It mimics how sound behaves when encountering obstacles like building partitions.

Optionally, call setRemoteAudioAttenuation or setPlayerAttenuation to set the sound attenuation properties for the user and media player, respectively, and specify whether to use this setting to forcefully override the sound attenuation factor in setZones.


_16
SpatialAudioZone mediaPlayerLeftZone = new SpatialAudioZone();
_16
mediaPlayerLeftZone.zoneSetId = 1;
_16
mediaPlayerLeftZone.audioAttenuation = 1f;
_16
float[] voicePosition = getVoicePosition(zoneTv);
_16
float[] viewRelativeSizeInAxis = getViewRelativeSizeInAxis(zoneTv);
_16
mediaPlayerLeftZone.position = new float[]{voicePosition[0], voicePosition[1], 0};
_16
mediaPlayerLeftZone.forward = new float[]{1.f, 0, 0};
_16
mediaPlayerLeftZone.right = new float[]{0, 1.f, 0};
_16
mediaPlayerLeftZone.up = new float[]{0, 0, 1.f};
_16
mediaPlayerLeftZone.forwardLength = viewRelativeSizeInAxis[1];
_16
mediaPlayerLeftZone.rightLength = viewRelativeSizeInAxis[0];
_16
mediaPlayerLeftZone.upLength = AXIS_MAX_DISTANCE;
_16
localSpatial.setZones(new SpatialAudioZone[]{mediaPlayerLeftZone});
_16
_16
localSpatial.setRemoteAudioAttenuation(uid, 0.5f, false);
_16
localSpatial.setPlayerAttenuation(mediaPlayer.getMediaPlayerId(), 0.5f, false);

Set headphone equalization (optional)

To optimize the audio experience, call setHeadphoneEQPreset to choose a preset headphone equalizer.

If the preset values do not provide the desired effect, call setHeadphoneEQParameters to self-adjust headphone equalization. After you execute this method, the preset values set by setHeadphoneEQPreset are overwritten.


_2
engine.setHeadphoneEQPreset(io.agora.rtc2.Constants.HEADPHONE_EQUALIZER_OVEREAR);
_2
engine.setHeadphoneEQParameters(10, 10);

Pause or turn off spatial audio

During a session, pause or turn off spatial audio as follows.

Pause spatial audio for a remote user

To disable a remote user's spatial audio, or to remove a remote user who has exited the channel, call removeRemotePosition to delete the user's spatial position information amd save computing resources.

caution

When a remote user leaves the channel, call removeRemotePosition to delete the user's spatial position information. Otherwise, the local user may not be able to hear the spatial audio from other remote users.

To restore the user's spatial audio, call updateRemotePosition to reset the remote user's position information.


_5
// Pause spatial audio for a specific remote user locally
_5
localSpatial.removeRemotePosition(uid);
_5
_5
// Resume spatial audio for a specific remote user locally
_5
localSpatial.updateRemotePosition(uid, getVoicePositionInfo(remoteLeftTv));

Pause spatial audio for all remote users

If you do not want to continue to experience local spatial audio, call clearRemotePositions to delete the spatial position information of all remote users.

caution

Calling this method prevents the local user from hearing the audio of all remote users. Agora recommends using this method with caution.

To resume hearing the remote user's audio later, call muteAllRemoteAudioStreams(false) to subscribe to the remote audio streams again.


_5
// Remove all remote users' spatial positions
_5
localSpatial.clearRemotePositions();
_5
_5
// Resume subscribing to remote users' audio streams
_5
localSpatial.muteAllRemoteAudioStreams(false);

Pause the local user's spatial audio

To unpublish the local audio stream, call muteLocalAudioStream(true).

To enable spatial audio again, call the same method with the parameter set to false.


_5
// Mute the local audio stream
_5
localSpatial.muteLocalAudioStream(true);
_5
_5
// Unmute the local audio stream
_5
localSpatial.muteLocalAudioStream(false);

Turn off spatial audio

To turn off spatial audio, call the enableSpatialAudio method of the RtcEngine instance and set the parameter to false. This resets all settings related to spatial audio.

To enable spatial audio again, call this method again and set the parameter to true, and then call the relevant APIs again to set the spatial audio effect.

Call the destroy method of the ILocalSpatialAudioEngine class to free up resources.


_5
// Disable spatial audio
_5
engine.enableSpatialAudio(false);
_5
_5
// Destroy the ILocalSpatialAudioEngine object
_5
ILocalSpatialAudioEngine.destroy();

Reference

This section contains content that completes the information on this page, or points you to documentation that explains other aspects to this product.

Voice Calling