9

在我的应用程序中,我想让用户在后台控制音频播放。我在 .plist 中设置了背景模式,并在 bg 中设置了我想要的播放模式。但是我无法通过触摸控制按钮得到任何响应。

我这样AudioSession设置

[[AVAudioSession sharedInstance] setCategory:AVAudioSessionCategoryPlayback error:nil];
[[AVAudioSession sharedInstance]setActive:YES error:nil];

然后,在放置我的播放器的 viewContoller 中,我beginReceivingRemoteControlEvents喜欢这样

 if ([[UIApplication sharedApplication] respondsToSelector:@selector(beginReceivingRemoteControlEvents)]){
    [[UIApplication sharedApplication] beginReceivingRemoteControlEvents];
    [[UIApplication sharedApplication] beginBackgroundTaskWithExpirationHandler:NULL];
    [self becomeFirstResponder];
    NSLog(@"Responds!");
}

它打印Responds!

但问题是这个方法永远不会被调用

    - (void)remoteControlReceivedWithEvent:(UIEvent *)event
{
    NSLog(@"Where is my event?");
    if(event.type == UIEventTypeRemoteControl)
    {
        switch (event.subtype) {
            case UIEventSubtypeRemoteControlTogglePlayPause:
                NSLog(@"Pause");
                [self playWords:playButton];
                break;
            case UIEventSubtypeRemoteControlNextTrack:
                NSLog(@"Next");
                [self next];
                break;
            case UIEventSubtypeRemoteControlPreviousTrack:
                NSLog(@"Prev");
                [self prev];
                break;

        }
    }

我什至试图写一个类别UIApplication让它成为第一响应者,但它没有帮助

@implementation UIApplication (RemoteEvents)
-(BOOL)canBecomeFirstResponder
{
    return YES;
}
@end

为什么会发生这种情况?

解决方案 这就是解决我的问题的方法在 iOS4 上进入后台播放音频

4

4 回答 4

13

我在我的项目中做了同样的工作,它工作正常。请关注这个,也许它会对你有所帮助。更改事件名称等。在我的代码中,_audio 是 AVAudioPlayer 的对象。

- (void)viewDidLoad {
    NSError *setCategoryErr = nil;
    NSError *activationErr  = nil;
    [[AVAudioSession sharedInstance] setCategory: AVAudioSessionCategoryPlayback error: &setCategoryErr];
    [[AVAudioSession sharedInstance] setActive: YES error: &activationErr];
}

- (void)viewWillAppear {

      [[UIApplication sharedApplication] beginReceivingRemoteControlEvents];
    [self becomeFirstResponder];
}


- (void)remoteControlReceivedWithEvent:(UIEvent *)event {
    switch (event.subtype) {
        case UIEventSubtypeRemoteControlPlay:
            [_audio play];
            break;
        case UIEventSubtypeRemoteControlPause:
            [_audio pause];
            break;
        default:
            break;
    }
}
于 2012-04-23T09:59:33.820 回答
4

有一种用于监听远程控制事件的更新机制。例如,当按下耳机播放/暂停按钮时执行一个块:

    MPRemoteCommandCenter *commandCenter = [MPRemoteCommandCenter sharedCommandCenter];

    [commandCenter.togglePlayPauseCommand addTargetWithHandler:^MPRemoteCommandHandlerStatus(MPRemoteCommandEvent * _Nonnull event) {
        NSLog(@"toggle button pressed");
        return MPRemoteCommandHandlerStatusSuccess;
    }];

或者,如果您更喜欢使用方法而不是块:

    [commandCenter.togglePlayPauseCommand addTarget:self action:@selector(toggleButtonAction)];

停止:

    [commandCenter.togglePlayPauseCommand removeTarget:self];

或者:

    [commandCenter.togglePlayPauseCommand removeTarget:self action:@selector(toggleButtonAction)];

您需要将其添加到文件的包含区域:

@import MediaPlayer;

要使其在后台运行,您必须将背景音频模式添加到应用程序的功能中。

于 2015-09-03T10:49:04.227 回答
1

查看 Apple 提供的此示例代码以获取使用示例。您的主视图控制器可能实际上并没有成为第一响应者。另一种用法是将此代码放在您的应用程序委托中(它将始终是第一个响应者)并在这些事件有机会传播并可能被其他响应者使用之前响应这些事件。

于 2012-05-05T23:49:54.333 回答
0

你有

respondsToSelector:@selector(beginReceivingRemoteControlEvents)]){

但是在您拥有的方法签名中

- (void)remoteControlReceivedWithEvent:(UIEvent *)event

您注册签名的地方是错误的。只需将其替换为

respondsToSelector:@selector(beginReceivingRemoteControlEvents:)]){

你错过了 : 这使应用程序将偶数发送到我假设的不存在的方法。我很惊讶您的应用程序没有崩溃。

于 2012-12-10T14:50:15.507 回答