1# Audio Playback Concurrency Policy 2 3## Audio Interruption Policy 4 5If multiple audio streams are played at the same time, the user may feel uncomfortable or even painful. To address this issue, OpenHarmony presets the audio interruption policy so that only the audio stream holding audio focus can be played. 6 7When an application attempts to play an audio, the system requests audio focus for the audio stream. The audio stream that gains the focus can be played. If the request is rejected, the audio stream cannot be played. If the audio stream is interrupted by another, it loses the focus and therefore the playback is paused. All these actions are automatically performed by the system and do not require additional operations on the application. However, to maintain state consistency between the application and the system and ensure good user experience, it is recommended that the application [listen for the audio interruption event](#listening-for-the-audio-interruption-event) and perform the corresponding processing when receiving such an event (specified by [InterruptEvent](../reference/apis/js-apis-audio.md#interruptevent9)). 8 9OpenHarmony presets two [audio interruption modes](#audio-interruption-mode) to specify whether audio concurrency is controlled by the application or system. You can choose a mode for each of the audio streams created by the same application. 10 11The audio interruption policy determines the operations (for example, pause, resume, duck, or unduck) to be performed on the audio stream. These operations can be performed by the system or application. To distinguish the body that executes the operations, the [audio interruption type](#audio-interruption-type) is introduced, and two audio interruption types are preset. 12 13### Audio Interruption Mode 14 15Two audio interruption modes, specified by [InterruptMode](../reference/apis/js-apis-audio.md#interruptmode9), are preset in the audio interruption policy: 16 17- **SHARED_MODE**: Multiple audio streams created by an application share one audio focus. The concurrency rules between these audio streams are determined by the application, without the use of the audio interruption policy. However, if another application needs to play audio while one of these audio streams is being played, the audio interruption policy is triggered. 18 19- **INDEPENDENT_MODE**: Each audio stream created by an application has an independent audio focus. When multiple audio streams are played concurrently, the audio interruption policy is triggered. 20 21The application can select an audio interruption mode as required. By default, the **SHARED_MODE** is used. 22 23You can set the audio interruption mode in either of the following ways: 24 25- If you [use the AVPlayer to develop audio playback](using-avplayer-for-playback.md), set the [audioInterruptMode](../reference/apis/js-apis-media.md#avplayer9) attribute of the AVPlayer to set the audio interruption mode. 26 27- If you [use the AudioRenderer to develop audio playback](using-audiorenderer-for-playback.md), call [setInterruptMode](../reference/apis/js-apis-audio.md#setinterruptmode9) of the AudioRenderer to set the audio interruption mode. 28 29 30### Audio Interruption Type 31 32The audio interruption policy (containing two audio interruption modes) determines the operation to be performed on each audio stream. These operations can be carried out by the system or application. To distinguish the executors, the audio interruption type, specified by [InterruptForceType](../reference/apis/js-apis-audio.md#interruptforcetype9), is introduced. 33 34- **INTERRUPT_FORCE**: The operation is performed by the system. The system forcibly interrupts audio playback. 35 36- **INTERRUPT_SHARE**: The operation is performed by the application. The application can take action or ignore as required. 37 38For the pause operation, the **INTERRUPT_FORCE** type is always used and cannot be changed by the application. However, the application can choose to use **INTERRUPT_SHARE** for other operations, such as the resume operation. The application can obtain the audio interruption type based on the value of the member variable **forceType** in the audio interruption event. 39 40During audio playback, the system automatically requests, holds, and releases the focus for the audio stream. When audio interruption occurs, the system forcibly pauses or stops playing or ducks the volume down for the audio stream, and sends an audio interruption event callback to the application. To maintain state consistency between the application and the system and ensure good user experience, it is recommended that the application [listen for the audio interruption event](#listening-for-the-audio-interruption-event) and perform processing when receiving such an event. 41 42For operations that cannot be forcibly performed by the system (for example, resume), the system sends the audio interruption event containing **INTERRUPT_SHARE**, and the application can choose to take action or ignore. 43 44## Listening for the Audio Interruption Event 45 46Your application are advised to listen for the audio interruption event when playing audio. When audio interruption occurs, the system performs processing on the audio stream according to the preset policy, and sends the audio interruption event to the application. 47 48Upon the receipt of the event, the application carries out processing based on the event content to ensure that the application state is consistent with the expected effect. 49 50You can use either of the following methods to listen for the audio interruption event: 51 52- If you [use the AVPlayer to develop audio playback](using-avplayer-for-playback.md), call [on('audioInterrupt')](../reference/apis/js-apis-media.md#onaudiointerrupt9) of the AVPlayer to listen for the event. 53 54- If you [use the AudioRenderer to develop audio playback](using-audiorenderer-for-playback.md), call [on('audioInterrupt')](../reference/apis/js-apis-audio.md#onaudiointerrupt9) of the AudioRenderer to listen for the event. 55 56 To deliver an optimal user experience, the application needs to perform processing based on the event content. The following uses the AudioRenderer as an example to describe the recommended application processing. (The recommended processing is similar if the AVPlayer is used to develop audio playback.) You can customize the code to implement your own audio playback functionality or application processing based on service requirements. 57 58```ts 59let isPlay; // An identifier specifying whether the audio stream is being played. In actual development, this parameter corresponds to the module related to the audio playback state. 60let isDucked; // An identifier specifying whether to duck the volume down. In actual development, this parameter corresponds to the module related to the audio volume. 61let started; // An identifier specifying whether the start operation is successful. 62 63async function onAudioInterrupt(){ 64 // The AudioRenderer is used as an example to describe how to develop audio playback. The audioRenderer variable is the AudioRenderer instance created for playback. 65 audioRenderer.on('audioInterrupt', async(interruptEvent) => { 66 // When an audio interruption event occurs, the audioRenderer receives the interruptEvent callback and performs processing based on the content in the callback. 67 // The audioRenderer reads the value of interruptEvent.forceType to see whether the system has forcibly performed the operation. 68 // The audioRenderer then reads the value of interruptEvent.hintType and performs corresponding processing. 69 if (interruptEvent.forceType === audio.InterruptForceType.INTERRUPT_FORCE) { 70 // If the value of interruptEvent.forceType is INTERRUPT_FORCE, the system has performed audio-related processing, and the application needs to update its state and make adjustments accordingly. 71 switch (interruptEvent.hintType) { 72 case audio.InterruptHint.INTERRUPT_HINT_PAUSE: 73 // The system has paused the audio stream (the focus is temporarily lost). To ensure state consistency, the application needs to switch to the audio paused state. 74 // Temporarily losing the focus: After the other audio stream releases the focus, the current audio stream will receive the audio interruption event corresponding to resume and automatically resume the playback. 75 isPlay = false; // A simplified processing indicating several operations for switching the application to the audio paused state. 76 break; 77 case audio.InterruptHint.INTERRUPT_HINT_STOP: 78 // The system has stopped the audio stream (the focus is permanently lost). To ensure state consistency, the application needs to switch to the audio paused state. 79 // Permanently losing the focus: No audio interruption event will be received. The user must manually trigger the operation to resume playback. 80 isPlay = false; // A simplified processing indicating several operations for switching the application to the audio paused state. 81 break; 82 case audio.InterruptHint.INTERRUPT_HINT_DUCK: 83 // The system has ducked the volume down (20% of the normal volume by default). To ensure state consistency, the application needs to switch to the volume decreased state. 84 // If the application does not want to play at a lower volume, it can select another processing mode, for example, proactively pausing the playback. 85 isDucked = true; // A simplified processing indicating several operations for switching the application to the volume decreased state. 86 break; 87 case audio.InterruptHint.INTERRUPT_HINT_UNDUCK: 88 // The system has restored the audio volume to normal. To ensure state consistency, the application needs to switch to the normal volume state. 89 isDucked = false; // A simplified processing indicating several operations for switching the application to the normal volume state. 90 break; 91 default: 92 break; 93 } 94 } else if (interruptEvent.forceType === audio.InterruptForceType.INTERRUPT_SHARE) { 95 // If the value of interruptEvent.forceType is INTERRUPT_SHARE, the application can take action or ignore as required. 96 switch (interruptEvent.hintType) { 97 case audio.InterruptHint.INTERRUPT_HINT_RESUME: 98 // The paused audio stream can be played. It is recommended that the application continue to play the audio stream and switch to the audio playing state. 99 // If the application does not want to continue the playback, it can ignore the event. 100 // To continue the playback, the application needs to call start(), and use the identifier variable started to record the execution result of start(). 101 await audioRenderer.start().then(async function () { 102 started = true; // Calling start() is successful. 103 }).catch((err) => { 104 started = false; // Calling start() fails. 105 }); 106 // If calling start() is successful, the application needs to switch to the audio playing state. 107 if (started) { 108 isPlay = true; // A simplified processing indicating several operations for switching the application to the audio playing state. 109 } else { 110 // Resuming the audio playback fails. 111 } 112 break; 113 default: 114 break; 115 } 116 } 117 }); 118} 119``` 120