Swift 中的音频播放未恢复到原始音量

Q.H*_*.H. 5 avaudioplayer ios avaudiosession swift

我正在尝试播放“叮”声以提醒用户发生事件。播放“叮”但背景音频(在本例中为默认 Music.App)在播放“叮”后没有恢复到其原始音量。但是,关闭应用程序后它将恢复正常音量。这就是我所拥有的:

这是我设置音频会话类别的地方:

     public override func viewDidLoad() {
            super.viewDidLoad()
            do {
                try AVAudioSession.sharedInstance().setCategory(AVAudioSessionCategoryPlayback, withOptions: AVAudioSessionCategoryOptions.DuckOthers)
                //refer to this link for swift's sound reference: https://developer.apple.com/ios/human-interface-guidelines/interaction/audio/
            } catch {
                print("unable to load audio session")
            }
            ....
     }
Run Code Online (Sandbox Code Playgroud)

这是我调用我的函数的地方:

if(!currentTimer.launchedNotification){
            playFinishedSound()
            AudioServicesPlayAlertSound(SystemSoundID(kSystemSoundID_Vibrate)) //handles phone vibration
            audioPlayerDidFinishPlaying(audioPlayer!, successfully: true)

        }
Run Code Online (Sandbox Code Playgroud)

这是我的私人功能:

private func playFinishedSound(){
        if let pathResource = NSBundle.mainBundle().pathForResource("Ding", ofType: "wav"){
            let soundToPlay = NSURL(fileURLWithPath: pathResource)
            do {
                audioPlayer = try AVAudioPlayer(contentsOfURL: soundToPlay)
                if(audioPlayer!.prepareToPlay()){
                    print("preparation success")
                    audioPlayer!.delegate = self
                    setAVAudioSession()
                    if(audioPlayer!.play()){
                        print("Sound play success")
                    }else{
                        print("Sound file could not be played")
                    }
                }else{
                    print("preparation failure")
                }

            }catch{
                print("Sound file could not be found")
            }
        }else{
            print("path not found")
        }
    }
Run Code Online (Sandbox Code Playgroud)

这是我设置音频会话的地方:

private func setAVAudioSession() {
        let session:AVAudioSession = AVAudioSession.sharedInstance()
        do{
            try session.setActive(true)
            print("session is active")
        }catch{
            print("could not make session active")
        }
    }
Run Code Online (Sandbox Code Playgroud)

这是 AVAudioPlayerDelegate 协议的委托方法:

public func audioPlayerDidFinishPlaying(player: AVAudioPlayer, successfully flag: Bool) {
        do {
            player.stop()
            try AVAudioSession.sharedInstance().setActive(false, withOptions: AVAudioSessionSetActiveOptions.NotifyOthersOnDeactivation)
            player.prepareToPlay()
            print("Session stopped successfully")
        }catch{
            print("Could not end audio session")
        }
    }
Run Code Online (Sandbox Code Playgroud)

同样,问题是如果后台有音乐播放,音乐音量会变弱,但是当音频会话处于非活动状态时,音乐音量不会恢复正常。

App*_*Guy 3

经过尝试后,我的解决方案效果很好。这里唯一需要注意的是,如果您尝试快速连续播放倒计时声音或其他声音,设备播放声音会有延迟,因此可能需要一些线程管理或其他方法来解决该问题。欢迎对此解决方案进行任何改进。

private let session = AVAudioSession.sharedInstance()

private func setSession(isActive: Bool) {
    if isActive {
        try! session.setCategory(AVAudioSession.Category.playback, options: AVAudioSession.CategoryOptions.duckOthers)
    } else {
        try! session.setCategory(AVAudioSession.Category.ambient, options: AVAudioSession.CategoryOptions.mixWithOthers)
    }
    try! self.session.setActive(isActive, options: .notifyOthersOnDeactivation)
}
Run Code Online (Sandbox Code Playgroud)

当您要播放、暂停或停止音频时,请调用此方法。

  1. 将类别设置为playback和 选项可duckOthers确保您的播放声音最大,而其他音频的音量则减小。

  2. 将类别设置为ambient和 选项可mixWithOthers确保您的播放与其他音频类型相同,从而导致音频恢复到播放前的音量。

  3. 最后将音频状态设置为活动或非活动( setActive)

注意事项:

使用try!当然不是推荐的做法,我提供了此代码作为让您滚动的方法。在适用的情况下实施最佳安全实践。