基于AVPlayer封装视频播放器(具有边下边播、离线缓存、自定义控制面板等功能)

最近因公司需求,要做一个类似QQ空间视频的轮播效果,故封装了一个功能齐全的视频播放器来实现项目的需求。现在我将封装过程中遇到的难题写出来和大家分享,以下只对重点进行说明,源码里有非常详细的注释,有兴趣的小伙伴可以下载参考。如果有此相似需求的小伙伴可以直接使用,在项目中播放视频只需简单两步。

    self.videoPlayer = [[LYVideoPlayer alloc] init];
    [self.videoPlayer playWithUrl:self.videoUrl showView:self.view];

本篇文章将从三个大的模块为大家介绍一个视频播放器的封装。

  • 第一:视频播放的实现;
  • 第二:离线缓存的实现;
  • 第三:自定义控制面板(自定义滑块可随意调整滑块大小和轨道高度、手势前进/后退、手势音量加减)。
    首先来看一下实现的效果:
播放器播放效果.gif
自定义滑块2.png
自定义滑块3.png

一、视频播放的实现

1、要现实视频的播放,得先知道在AVFoundation框架下的三个类:AVPlayerItem、AVPlayer、AVPlayerLayer。AVPlayerItem是一个媒体资源管理类,负责数据的获取与分发;AVPlayer负责解码数据;AVPlayerLayer 是图层显示,用于数据的展示。

    //1.创建播放器
    self.currentPlayerItem = [AVPlayerItem playerItemWithURL:url];
    self.player = [AVPlayer playerWithPlayerItem:self.currentPlayerItem];
    self.currentPlayerLayer = [AVPlayerLayer playerLayerWithPlayer:self.player];

2、第一步已经创建好了播放器,接下来就是去播放了,那么问题就来了,咱们怎么知道什么时候可以开始播放了呢?这就需要去监听播放器的状态了,通过KVO监听AVPlayerItem的状态,获得状态后就可以去让AVPlayer执行播放的方法了。


    //1.通过KVO监听AVPlayerItem的状态
    [self.currentPlayerItem addObserver:self forKeyPath:@"status" options:NSKeyValueObservingOptionNew context:nil];
  
    //2.方法回调-根据状态做相应的逻辑处理
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context{
    
    AVPlayerItem *playerItem = (AVPlayerItem *)object;
    
    if ([keyPath isEqualToString:@"status"]) {
        AVPlayerItemStatus status = playerItem.status;
        switch (status) {
            case AVPlayerItemStatusUnknown:{
                NSLog(@"======== 播放失败");
            }
                break;
                
            case AVPlayerItemStatusReadyToPlay:{
                NSLog(@"========= 准备播放");
                //去播放
                [self play];
                //图层显示
                [self handleShowViewSublayers];
            }
                break;
                
            case AVPlayerItemStatusFailed:{
                NSLog(@"======== 播放失败");
            }
                break;
                
            default:
                break;
        }
    }
}

3、视频的播放很简单,但是只有播放功能还远远不能满足咱们的需求啊!好吧,继续来。其中比较伤脑筋的是菊花(此菊花非彼菊花~)的显示逻辑,有多伤脑筋我就不说太细了,反正我相信做过这个的小伙伴应该是明白菊花带来的伤痛的。首先菊花的显示第一次肯定是在加载数据的时候进行显示的,再者就是在播放到没有缓冲数据的时候进行显示,这个很容易实现。实现方法就是利用AVPlayerItem进行监听,代码如下

 //监听到当前没有缓冲数据   
 [self.currentPlayerItem addObserver:self forKeyPath:@"playbackBufferEmpty" options:NSKeyValueObservingOptionNew context:nil];

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context{
    
    AVPlayerItem *playerItem = (AVPlayerItem *)object;
    if ([keyPath isEqualToString:@"playbackBufferEmpty"]) {
        self.isPlaying = NO;
        self.isBufferEmpty = YES;
        self.lastBufferValue = self.currentBufferValue;
        [self.videoPlayControl videoPlayerDidLoading];//显示菊花
        
        NSLog(@"====playbackBufferEmpty");
    }
}

好了,伤脑筋的终于来了,菊花显示了该什么时候去让它消失呢?有的小伙伴可能会说利用AVPlayer进行播放的监听啊,当视频播放的时候就让菊花消失就行了。当然这是肯定的,会在这里监听做事情,但是事情远不止这一点。比如当正在缓冲的时候被暂停了,那么监听视频的播放来让菊花消失肯定是不能满足需求的,还有一个非常蛋疼的问题就是当拖动滑块到没有缓冲的地方的时候,这时候明明正在缓冲数据没有播放,但是这个时候莫名其妙的就还会来到这个地方,所以我不得已在代码里做了一些不人性化的操作,就是获取当前的本地时间,然后在拖动滑块的时候记录下当前时间来和下次进入这个方法的时候作对比,如果是时间差大于1秒以上的一般就是真正的在播放了。


- (void)addObserver {
    //监听播放进度
    __weak typeof(self) weakSelf = self;
    self.timeObserve = [self.player addPeriodicTimeObserverForInterval:CMTimeMake(1, 1) queue:dispatch_get_main_queue() usingBlock:^(CMTime time) {
        CGFloat current = CMTimeGetSeconds(time);
        CGFloat total = CMTimeGetSeconds(weakSelf.currentPlayerItem.duration);
        CGFloat progress = current / total;
        
        weakSelf.videoPlayControl.currentTime = current;
        weakSelf.videoPlayControl.playValue = progress;
        
        /***** 这里是比较蛋疼的,当拖动滑块到没有缓冲的地方并且没有开始播放时,也会走到这里 *************/
        if (weakSelf.isCanToGetLocalTime) {
             weakSelf.localTime = [weakSelf getLocalTime];
        }
        NSInteger timeNow = [weakSelf getLocalTime];
        if (timeNow - weakSelf.localTime > 1.5) {
            [weakSelf.videoPlayControl videoPlayerDidBeginPlay];
            weakSelf.isCanToGetLocalTime = YES;
        }
    }];
}

二、离线缓存的实现

实现数据的离线缓存,我的做法是,当在建立起数据请求的时候,根据url生成一个文件路径,让数据下载到一个临时的文件路径下。第一种情况:当请求发起时一直下载到下载成功,这时候就将该文件移动到缓存目录下缓存起来。第二种情况:当中断下载数据时,对该临时文件不做任何处理,然后再次播放该视频请求数据时,根据url生成的路径查找当前的临时路径下有无该文件,如果有说明该文件没有下载完成,则需要读到这个文件然后做断点续传操作,让该文件继续下载,而不是重头开始下载。我在这里是提供了一个离线缓存的思路,如想深入研究离线缓存和断点下载的小伙伴可以去这里看看【补充】NSURLSession 详解离线断点下载的实现


- (void)fileJudge{
    //判断当前目录下有无已有下载的临时文件
    if ([_fileManager fileExistsAtPath:self.videoTempPath]) {
        //存在已下载数据的文件
        _fileHandle = [NSFileHandle fileHandleForUpdatingAtPath:self.videoTempPath];
        _curruentLength = [_fileHandle seekToEndOfFile];
        
    }else{
        //不存在文件
        _curruentLength = 0;
        //创建文件
        [_fileManager createFileAtPath:self.videoTempPath contents:nil attributes:nil];
        _fileHandle = [NSFileHandle fileHandleForUpdatingAtPath:self.videoTempPath];
    }
    //发起请求
    [self sendHttpRequst];
}

//网路请求方法
- (void)sendHttpRequst
{
    [_fileHandle seekToEndOfFile];
    NSURL *url = [NSURL URLWithString:_videoUrl];
    NSMutableURLRequest *requeset = [NSMutableURLRequest requestWithURL:url];
    
    //指定头信息  当前已下载的进度
    [requeset setValue:[NSString stringWithFormat:@"bytes=%ld-", _curruentLength] forHTTPHeaderField:@"Range"];
    
    //创建请求
    NSURLSessionDataTask *dataTask = [self.session dataTaskWithRequest:requeset];
    self.dataTask = dataTask;
    
    //发起请求
    [self.dataTask resume];
}

-(void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error {
    
    if (error == nil) { //下载成功
        //当前下载文件的临时路径
        NSURL *tempPathURL = [NSURL fileURLWithPath:self.videoTempPath];
        //缓存路径
        NSURL *cachefileURL = [NSURL fileURLWithPath:self.videoCachePath];

        // 如果没有该文件夹,创建文件夹
        if (![self.fileManager fileExistsAtPath:self.videoCachePath]) {
            [self.fileManager createDirectoryAtPath:self.videoCachePath withIntermediateDirectories:YES attributes:nil error:nil];
        }
        
        // 如果该路径下文件已经存在,就要先将其移除,在移动文件
        if ([self.fileManager fileExistsAtPath:[cachefileURL path] isDirectory:NULL]) {
            [self.fileManager removeItemAtURL:cachefileURL error:NULL];
        }
        //移动文件至缓存目录
        [self.fileManager moveItemAtURL:tempPathURL toURL:cachefileURL error:NULL];
    }
}

三、自定义控制面板

该控制面板具有一个视频播放器应该具备的基本功能:播放与暂停、滑块拖动播放、显示视频当前播放时间和总的时间。另外我增加了一些手势的操作:左右滑动实现前进和后退,上下滑动实现音量的加减,单击实现面板的显示与收起。
因为系统的滑块UISlider滑块控件要说实用倒也是能用,但是要改成自己想要的UI那也是件蛋疼的事情,所以我专门对滑块又做了一次单独的封装,封装好的滑块控件LYSlider,想要改变其高度和大小只需要改变相应的两个属性(trackHeight、thumbVisibleSize)就行了,当然想到系统的滑块和进度条是两个分开的控件,在此我也将进度条一起封装进去了,也就是滑块里具有缓冲进度条的功能,只要你对bufferProgress这个属性传值,那么这个进度条就会显示出来了。这里就上LYSlider初始化时候的代码了,想要一谈究竟的小伙伴就去我的github(地址在最后)下载源码吧!码字不容易写代码更不容易!记得给我star哦~


- (LYSlider *)videoSlider{
    if (!_videoSlider) {
        _videoSlider = [[LYSlider alloc] initWithFrame:CGRectMake(CGRectGetMaxX(self.currentLabel.frame) + 5, 0, _frame.size.width - CGRectGetMaxX(self.currentLabel.frame) - self.totalLabel.frame.size.width - 20 , BottomHeight)];
        
        //设置滑块图片样式
        // 1 通过颜色创建 Image
         UIImage *normalImage = [UIImage createImageWithColor:[UIColor redColor] radius:5.0];
        
        // 2 通过view 创建 Image
        UIView *highlightView = [[UIView alloc] initWithFrame:CGRectMake(0, 0, 12, 12)];
        highlightView.layer.cornerRadius = 6;
        highlightView.layer.masksToBounds = YES;
        highlightView.backgroundColor = [UIColor redColor];
        UIImage *highlightImage = [UIImage creatImageWithView:highlightView];

        [_videoSlider setThumbImage:normalImage forState:UIControlStateNormal];
        [_videoSlider setThumbImage:highlightImage forState:UIControlStateHighlighted];
        
        _videoSlider.trackHeight = 1.5;    //设置轨道高度
        _videoSlider.thumbVisibleSize = 12;//设置滑块(可见的)大小
        
        [_videoSlider addTarget:self action:@selector(sliderValueChange:) forControlEvents:UIControlEventValueChanged];//正在拖动
        [_videoSlider addTarget:self action:@selector(sliderTouchEnd:) forControlEvents:UIControlEventEditingDidEnd];//拖动结束
        [self.bottomView addSubview:_videoSlider];
    }
    return _videoSlider;
}
最后的话

本篇文章只是对播放器的简单的封装,如有不合理的地方还望指正!如果你看了这篇文章对你有些许的帮助,我也将感到非常荣幸!也请点击下方的喜欢或关注本人

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

推荐阅读更多精彩内容