繁体   English   中英

音频会话中断结束后,AVAudioRecorder 未在后台录制

[英]AVAudioRecorder not recording in background after audio session interruption ended

我正在我的应用程序中录制音频,包括前台和后台。 我还处理 AVAudioSessionInterruptionNotification 以在中断开始时停止录制并在中断结束时重新开始。 虽然在前台它按预期工作,但当应用程序在后台录音并且我接到电话时,它不会在通话结束后再次开始录音。 我的代码如下:

        - (void)p_handleAudioSessionInterruptionNotification:(NSNotification *)notification
        {
            NSUInteger interruptionType = [[[notification userInfo] objectForKey:AVAudioSessionInterruptionTypeKey] unsignedIntegerValue];

            if (interruptionType == AVAudioSessionInterruptionTypeBegan) {
                if (self.isRecording && !self.interruptedWhileRecording) {

                    [self.recorder stop];

                    self.interruptedWhileRecording = YES;
                    return;
                }
            }

            if (interruptionType == AVAudioSessionInterruptionTypeEnded) {
                if (self.interruptedWhileRecording) {
                    NSError *error = nil;
                    [[AVAudioSession sharedInstance] setActive:YES error:&error];

                    NSDictionary *settings = @{
                                       AVEncoderAudioQualityKey: @(AVAudioQualityMax),
                                       AVSampleRateKey: @8000,
                                       AVFormatIDKey: @(kAudioFormatLinearPCM),
                                       AVNumberOfChannelsKey: @1,
                                       AVLinearPCMBitDepthKey: @16,
                                       AVLinearPCMIsBigEndianKey: @NO,
                                       AVLinearPCMIsFloatKey: @NO
                                       };

                    _recorder = [[AVAudioRecorder alloc] initWithURL:fileURL settings:settings error:nil];

                    [self.recorder record];

                    self.interruptedWhileRecording = NO;
                    return;
                }
            }
        }

请注意, fileURL 指向 NSDocumentDirectory 子目录中的新 caf 文件。 背景模式音频已配置。 我也尝试过 voip 和播放静音,但都没有成功。

AVAudioSessionInterruptionTypeEnded 块中的 NSError 是一个 OSStatus 错误 560557684,我还没有找到如何解决。

任何帮助将非常感激。

错误 560557684 是针对AVAudioSessionErrorCodeCannotInterruptOthers 当您的后台应用程序尝试激活不与其他音频会话混合的音频会话时,就会发生这种情况。 后台应用程序无法启动不与前台应用程序的音频会话混合的音频会话,因为这会中断用户当前正在使用的应用程序的音频。

要解决此问题,请确保将会话类别设置为可混合的类别,例如AVAudioSessionCategoryPlayback 还要确保设置类别选项AVAudioSessionCategoryOptionMixWithOthers (必需)和AVAudioSessionCategoryOptionDuckOthers (可选)。 例如:

// background audio *must* mix with other sessions (or setActive will fail)
NSError *sessionError = nil;
[[AVAudioSession sharedInstance] setCategory:AVAudioSessionCategoryPlayback
                                 withOptions:AVAudioSessionCategoryOptionMixWithOthers | AVAudioSessionCategoryOptionDuckOthers
                                       error:&sessionError];
if (sessionError) {
    NSLog(@"ERROR: setCategory %@", [sessionError localizedDescription]);
}

错误代码 560557684 实际上是 32 位整数中的 4 个 ascii 字符 '!int'。 错误代码列在 AVAudioSession.h 文件中(另请参阅AVAudioSession ):

    @enum AVAudioSession error codes
    @abstract   These are the error codes returned from the AVAudioSession API.
...
    @constant   AVAudioSessionErrorCodeCannotInterruptOthers
        The app's audio session is non-mixable and trying to go active while in the background.
        This is allowed only when the app is the NowPlaying app.

typedef NS_ENUM(NSInteger, AVAudioSessionErrorCode)
{
...
    AVAudioSessionErrorCodeCannotInterruptOthers = '!int', /* 0x21696E74, 560557684 */
...

我添加了以下内容

[[UIApplication sharedApplication] beginReceivingRemoteControlEvents];

在配置 AVAudioSession 之前,它起作用了。 还不知道会出现什么bug。

当我的应用程序在后台时尝试使用AVSpeechSynthesizer().speak()时,我遇到了同样的错误。 @progmr 的回答为我解决了这个问题,尽管我也不得不调用AVAudioSession.sharedInstance().setActive(true) 为了完整起见,这是我在Swift 5 中的代码。

application(_:didFinishLaunchingWithOptions:)

do {
  try AVAudioSession.sharedInstance().setCategory(AVAudioSession.Category.playback,
                                                  options: [.mixWithOthers,
                                                            .duckOthers])

} catch let error as NSError {
  print("Error setting up AVAudioSession : \(error.localizedDescription)")
}

然后在我的视图控制器中:

do {
  try AVAudioSession.sharedInstance().setActive(true)
} catch let error as NSError {
  print("Error : \(error.localizedDescription)")
}

let speechUtterance = AVSpeechUtterance(string: "Hello, World")
let speechSynth = AVSpeechSynthesizer()
speechSynth.speak(speechUtterance)

注意:当setActive(true)被调用时,它会降低当时正在播放的其他任何东西的音量。 之后要调高音量,您需要调用setActive(false) - 对我来说,最好的时间是在我收到通知说演讲已在相应的AVSpeechSynthesizerDelegate方法中结束时。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM