AVPlayer循环播放音频、锁屏界面远程控制、

AVPlayer可以用来播放音频和视频,今天要说的就是用AVPlayer的一个子类 AVQueuePlayer来播放音频、这个类可以用来播放队列,当前AVPlayerItem播放完了,如果nextItem存在就会自动播放下一个Item,比AVPlayer要方便一点。

一、后台播放设置
先导入两个框架

#import <AVFoundation/AVFoundation.h>
#import <MediaPlayer/MediaPlayer.h>

设置后台播放

   AVAudioSession *session = [AVAudioSession sharedInstance];
    [session setActive:YES error:nil];
    [session setCategory:AVAudioSessionCategoryPlayback error:nil];
屏幕快照 2017-08-24 下午2.09.25.png

二、AVQueuePlayer初始化

-(void)initPlayer{
    //准备数据 一个本地MP3和一个URL MP3
    AVPlayerItem*firstItem = [AVPlayerItem playerItemWithURL:[NSURL fileURLWithPath: [[NSBundle mainBundle] pathForResource:@"那些花儿" ofType:@"mp3"]]];
    AVPlayerItem*secondItem = [AVPlayerItem playerItemWithURL:[NSURL URLWithString:@"http://sc1.111ttt.com/2015/1/06/06/99060941326.mp3"]];
    
    SongModel*firstModel = [[SongModel alloc]init];
    firstModel.songName = @"那些花儿";
    firstModel.singer = @"朴树";
    firstModel.picture = [UIImage imageNamed:@"pushu.jpg"];
    firstModel.duration = 294;
    firstModel.item = firstItem;
    
    SongModel*secondModel = [[SongModel alloc]init];
    secondModel.songName = @"演员";
    secondModel.singer = @"薛之谦";
    secondModel.picture = [UIImage imageNamed:@"xue.png"];
    secondModel.duration = 262;
    secondModel.item = secondItem;
    _songInfoArray = @[firstModel,secondModel];
    
//这里初始化设置一个或多个都是可以的,这里我设置一个的原因主要是上一曲和下一曲的时候会清空items
    self.player = [[AVQueuePlayer alloc]initWithItems:@[firstItem]];
    
    _index = 0;
// item和player都有status 属性 通常我们观察item的status 是因为可以检测资源是否可以播放,当然这里直接调用play方法也是可以直接播放的
    [firstItem addObserver:self forKeyPath:@"status" options:NSKeyValueObservingOptionNew context:nil];
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(playeyEnd:) name:AVPlayerItemDidPlayToEndTimeNotification object:nil];
}

初始化完成之后直接调用[AVPlayer play]也是可以直接播放的,但是资源出现问题的话还是需要手动处理的所以我们在这里对Item添加了一个观察者,并对AVPlayer播放完成添加了一个通知。

通知和观察者模式实现代码

///播放结束
-(void)playeyEnd:(NSNotification*)notify{
    NSLog(@"end");
    [self nextSong];
    
}

///AVPlayerItem observer
-(void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context{
    AVPlayerItem*item = (AVPlayerItem*)object;
    if ([keyPath isEqualToString:@"status"]) {
        if (item.status==AVPlayerItemStatusReadyToPlay) {
            NSLog(@"play");
            
            [self setLockScreenNowPlayingInfo];
            
            [item removeObserver:self forKeyPath:@"status"];
            
            [self.player play];

        }
        
        if (item.status==AVPlayerItemStatusFailed) {
            
            NSLog(@"filad");
            
            [self setLockScreenNowPlayingInfo];
            
            [item removeObserver:self forKeyPath:@"status"];
            
            [self nextSong];
        }
    }
}

切歌控制

#warning 在同一时间内一个item只能占用一个位置、所以这里是先删除,再添加
-(void)nextSong{
    //下一首
    if (_index==_songInfoArray.count-1) {
        _index =0;
    }else{
        _index++;
    }
    SongModel*model = _songInfoArray[_index];
    [model.item addObserver:self forKeyPath:@"status" options:NSKeyValueObservingOptionNew context:nil];
    [self.player removeAllItems];
    [self.player insertItem:model.item afterItem:nil];
    [self.player seekToTime:kCMTimeZero];
    
    [self setLockScreenNowPlayingInfo]; 
}

-(void)lastSong{
    //上一首
    if (_index==0) {
        _index =_songInfoArray.count-1;
        
    }else{
        _index--;  
    }
    SongModel*model = _songInfoArray[_index];
    [model.item addObserver:self forKeyPath:@"status" options:NSKeyValueObservingOptionNew context:nil];
    [self.player removeAllItems];
    [self.player insertItem:model.item afterItem:nil];
    [self.player seekToTime:kCMTimeZero];
    [self setLockScreenNowPlayingInfo];
}

这里有一个小坑、AVQueuePlayer提供了跳转到下一个Item的方法

- (void)advanceToNextItem;

却没有提供LastItem的方法,所以要实现LastItem的方法我是通过来实现的

[self.player insertItem:model.item afterItem:nil]; 

特别提醒在同一时间同一个Item只能加入一次,所以在insert之前需要将之前的Item删除掉、我选择了删除全部、这样同一时间PlayerQueue中就只存在一个Item了、我觉得这样更容易控制一点。而关于 [self.player seekToTime:kCMTimeZero]; 这个是为了让Item上一次的播放时间置零,如果不置零就会继续播放(感觉这个功能播放视频的时候这个还是比较好用的);

三、注册远程控制(锁屏界面控制和耳机控制)

需要先让应用能够接受远程控制,并成为第一响应者

屏幕快照 2017-08-24 下午2.11.29.png
#pragma mark - 远程控制接收方法的设置
- (void)remoteControlReceivedWithEvent:(UIEvent *)event {
    if (event.type == UIEventTypeRemoteControl) {  //判断是否为远程控制
        switch (event.subtype) {
            case  UIEventSubtypeRemoteControlPlay:
            {
                if (!_isPlaying) {
                    [self.player play];
                }
                _isPlaying=!_isPlaying;
            }
                break;
            case UIEventSubtypeRemoteControlPause:
            {
                if (_isPlaying) {
                    [self.player pause];
                }
                _isPlaying = !_isPlaying;
            }
                break;
            case UIEventSubtypeRemoteControlNextTrack:
            {
                [self nextSong];
            }
                break;
            case UIEventSubtypeRemoteControlPreviousTrack:
            {
                [self lastSong];
            }
                break;
            default:
                break;
        }
    }
}

四、设置锁屏信息
期间我看了网易云音乐的锁屏效果、设置了lyrics,结果我这里设置了并没有什么效果。。。

///设置锁屏信息
- (void)setLockScreenNowPlayingInfo
{
    SongModel*model = self.songInfoArray[_index];
    //更新锁屏时的歌曲信息
    MPMediaItemArtwork *artWork = [[MPMediaItemArtwork alloc] initWithImage:model.picture];
    
    NSDictionary *dic = @{MPMediaItemPropertyTitle:model.songName,
                          MPMediaItemPropertyArtist:model.singer,
                          //                          MPMediaItemPropertyLyrics:@"hello lyrics break ",
                          //                          MPMediaItemPropertyReleaseDate:@"2017-08-23",//唱片发布日期
                          MPMediaItemPropertyPlaybackDuration:@(model.duration),//设置锁屏界面歌曲时间
                          MPMediaItemPropertyArtwork:artWork//锁屏界面图片
                          };
    [[MPNowPlayingInfoCenter defaultCenter] setNowPlayingInfo:dic];
}

最后附上真机锁屏的效果以及Demo地址
https://github.com/LuoCongMing/AVQueuePlayerDemo.git

IMG_1878.PNG
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,458评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,454评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,171评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,062评论 0 207
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,440评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,661评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,906评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,609评论 0 200
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,379评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,600评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,085评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,409评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,072评论 3 237
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,088评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,860评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,704评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,608评论 2 270

推荐阅读更多精彩内容