RemoteControl,这里就翻译为远程控制吧。
远程控制是为用户提供操作App多媒体的。远程控制事件源于外部附件或由系统显示的传输控制,并通过媒体播放器框架的类传送到应用程序。播放音频或视频内容的应用程序使用这些事件来开始和停止播放,更改曲目,甚至速度的项目。所有的媒体应用程序应该支持这些事件。
除了支持远程控制事件,应用程序可以使用媒体播放器框架,以提供播放信息的曲目。该系统在适当的地方显示播放信息,如锁屏和控制中心。有关媒体播放器框架类的更多信息,见媒体播放器框架参考 Media Player Framework Reference .
RemoteControl可以用来在不打开app的情况下控制app中的多媒体播放,主要包括:
要接收远程控制事件,需要做到以下的几点:
若应用程序还提供了正在播放的信息,使用 MPNowPlayingInfoCenter
对象在适当的时候更新信息。
要使App支持Remote Control,需要设置一下info.plist文件,添加required background modes,并添加一个值为:
App playsaudioorstreamsaudio/videousingAirPlay
接下来,播放音频之前先要设置AVAudioSession模式:
AVAudioSession *session = [AVAudioSession sharedInstance]; [sessionsetCategory:AVAudioSessionCategoryPlayAndRecord withOptions:AVAudioSessionCategoryOptionDefaultToSpeaker error:nil];
在iOS上,响应者链如上图,由图可知,若其它视图没有处理,最终会传到UIApplication,因此使用UIApplication作为第一响应者是最合适的。
扩展头文件声明,我们在监听到对应的事件时,发送通知。
// // UIApplication+RemoteControl.h // IOSAudioRemoteControl // // Created by huangyibiao on 15/3/25. // Copyright (c) 2015年 huangyibiao. All rights reserved. // #import <UIKit/UIKit.h> // 播放 extern NSString *kRemoteControlPlayTapped; // 暂停 extern NSString *kRemoteControlPauseTapped; // 停止 extern NSString *kRemoteControlStopTapped; // 前一首 extern NSString *kRemoteControlPreviousTapped; // 后一首 extern NSString *kRemoteControlNextTapped; // 其它 extern NSString *kRemoteControlOtherTapped; @interface UIApplication (RemoteControl) /** * 注册对remote control事件的监听 * * @param observer 监听者 * @param selector 回调 */ - (void)observeRemoteControl:(id)observerselector:(SEL)selector; @end
在实现文件中,我们需要处理监听:
// // UIApplication+RemoteControl.m // IOSAudioRemoteControl // // Created by huangyibiao on 15/3/25. // Copyright (c) 2015年 huangyibiao. All rights reserved. // #import "UIApplication+RemoteControl.h" const NSString *kRemoteControlPlayTapped = @"kRemoteControlPlayTapped"; const NSString *kRemoteControlPauseTapped = @"kRemoteControlPauseTapped"; const NSString *kRemoteControlStopTapped = @"kRemoteControlStopTapped"; const NSString *kRemoteControlPreviousTapped = @"kRemoteControlForwardTapped"; const NSString *kRemoteControlNextTapped = @"kRemoteControlBackwardTapped"; const NSString *kRemoteControlOtherTapped = @"kRemoteControlOtherTapped"; @implementation UIApplication (RemoteControl) - (BOOL)canBecomeFirstResponder { return YES; } - (void)remoteControlReceivedWithEvent:(UIEvent *)event { switch (event.subtype) { caseUIEventSubtypeRemoteControlPlay: [selfpostNotification:kRemoteControlPlayTapped]; break; caseUIEventSubtypeRemoteControlPause: [selfpostNotification:kRemoteControlPauseTapped]; break; caseUIEventSubtypeRemoteControlStop: [selfpostNotification:kRemoteControlStopTapped]; break; caseUIEventSubtypeRemoteControlNextTrack: [selfpostNotification:kRemoteControlNextTapped]; break; caseUIEventSubtypeRemoteControlPreviousTrack: [selfpostNotification:kRemoteControlPreviousTapped]; break; default: [selfpostNotification:kRemoteControlOtherTapped]; break; } } - (void)postNotification:(const NSString *)notificationName { [[NSNotificationCenter defaultCenter] postNotificationName:(NSString *)notificationNameobject:nil]; } - (void)observeRemoteControl:(id)observerselector:(SEL)selector { NSNotificationCenter *center = [NSNotificationCenter defaultCenter]; [centeraddObserver:observerselector:selectorname:(NSString *)kRemoteControlNextTappedobject:nil]; [centeraddObserver:observerselector:selectorname:(NSString *)kRemoteControlPauseTappedobject:nil]; [centeraddObserver:observerselector:selectorname:(NSString *)kRemoteControlStopTappedobject:nil]; [centeraddObserver:observerselector:selectorname:(NSString *)kRemoteControlPreviousTappedobject:nil]; [centeraddObserver:observerselector:selectorname:(NSString *)kRemoteControlPlayTappedobject:nil]; [centeraddObserver:observerselector:selectorname:(NSString *)kRemoteControlOtherTappedobject:nil]; } @end
在播放音频的界面,可以是在UIViewController里,也只可以是某个UIView里,因此在对应的类在创建或者初始化时调用监听:
UIApplication *app = [UIApplication sharedApplication]; [appobserveRemoteControl:self selector:@selector(onRemoteControleStateChanged:)];
然后处理一下事件回调:
#pragma mark - remote control - (void)onRemoteControleStateChanged:(NSNotification *)notification { if ([notification.nameisEqualToString:kRemoteControlPlayTapped]) { [self play]; } else if ([notification.nameisEqualToString:kRemoteControlPauseTapped]) { [self pause]; } else if ([notification.nameisEqualToString:kRemoteControlNextTapped]) { [self playNext]; } else if ([notification.nameisEqualToString:kRemoteControlPreviousTapped]) { } else if ([notification.nameisEqualToString:kRemoteControlStopTapped]) { [self_requestPauseAudioPlaying:nil]; } else if ([notification.nameisEqualToString:kRemoteControlOtherTapped]) { } }
要想在当前播放中心中显示当前播放的音频的信息,需要通过 MPNowPlayingInfoCenter
类,它是一个单例。我们要显示什么就在它的 nowPlayingInfo
属性中设置相关配置数据。
要使用这个类及其相关属性定义,需要引入这两个头文件:
#import <MediaPlayer/MPNowPlayingInfoCenter.h> #import <MediaPlayer/MPMediaItem.h>
这里只设置几项,更多设置,请到MPMediaItem.h头文件中查看:
NSMutableDictionary *dict = [[NSMutableDictionary alloc]init]; // 设置歌曲播放时长 if (self.time != nil && self.time.length != 0) { [dictsetObject:@(self.time.doubleValue)forKey:MPMediaItemPropertyPlaybackDuration]; } // 设置歌曲名称 if (self.title != nil && self.title.length != 0) { [dictsetObject:self.titleforKey:MPMediaItemPropertyTitle]; } // 设置锁屏时的图片 if (self.iconView.image && [self.iconView.imageisKindOfClass:[UIImage class]]) { MPMediaItemArtwork *artwork = [[MPMediaItemArtwork alloc]initWithImage:self.iconView.image]; [dictsetObject:artworkforKey:MPMediaItemPropertyArtwork]; } [MPNowPlayingInfoCenter defaultCenter].nowPlayingInfo = dict;
到此就可以实现我们远程控制的功能。不过这里只考虑实现,并不考虑什么时候接收,什么时候不再接收,而这里的做法时什么时候都接收远程控制。这此事就留给大家去学习吧!
大家可以到我的GITHUB下载demo: https://github.com/CoderJackyHuang/IOSAudioRemoteControl
苹果官方文档:Remote Control Events
如果在使用过程中遇到问题,或者想要与我交流,可加入有问必答 QQ群: 324400294
关注微信公众号: iOSDevShares
关注新浪微博账号:标哥Jacky
标哥的GITHUB地址: CoderJackyHuang
如果您觉得文章对您很有帮忙,希望得到您的支持。您的捐肋将会给予我最大的鼓励,感谢您的支持!
支付宝捐助 | 微信捐助 |
---|---|