AVPlayer 实现视频播放总结

因为产品需求要修改视频播放的展示策略,就简单的梳理了一下目前端上的视频播放功能,简单整理了一下基本实现,以便于之后查阅。

播放器实现思路

由于视频的展示在不同的产品上会有不同的样式,比如列表页的视频样式,具体详情页视频的样式以及点击视频放大播放的样式都是不同的,因此最好把具体的视频播放逻辑与展示 UI 分开写,便于以后的功能添加及修改。


C7198711-CD8E-446C-B4C0-0F9178084E94.png

AVPlayer 实现视频播放

  • AVPlayer : AVPlayer 提供了单个视频播放功能,可以播放本地视频和网络资源,提供播放,暂停等功能。
  • AVPlayerItem : 一个媒体资源管理对象,管理者视频的一些基本信息和状态
  • AVPlayerLayer : 是 CALayer 的子类,AVPlayer 实例化的对象(视频内容)需要需要放到 AVPlayerLayer 上才能播放。

视频播放功能的实现大致需要以下几个步骤:
1、创建 AVPlayer 实例,并将其添加到 AVPlayerLayer 实例上;
2、监听 AVPlayerItem 的 status 属性状态变化,判断视频是否可以正常播放;
3、监听 AVPlayerItem 的 loadedTimeRanges 属性状态变化,处理下载进度条显示;
4、调用 addPeriodicTimeObserverForInterval:queue:usingBlock: 方法来处理视频播放进度条的变化;
5、调用- (void)seekToTime: toleranceBefore: toleranceAfter: 方法,实现视频跳转到某一时刻播放
6、 视频播放结束处理

以下是具体实现过程:
创建 AVPlayerItem 对象,并通过该对象实例化 AVPlayer 对象,将 AVPlayer 添加到 AVPlayerLayer 对象上

    self.playerItem = [AVPlayerItem playerItemWithURL:videoUrl];
    self.player = [[AVPlayer alloc] initWithPlayerItem:self.playerItem];
    [self.playerLayer setPlayer:self.player];

以上对象都创建完了并不能顺利播放视频,需要知道视频是否下载成功,能不能顺利播放,网络不好或者链接无效的情况下我们都应该对其作出不同处理,这里就需要添加 KVO 监听视频的缓存进度和播放状态,并且注意要在适当的时候移除。

    // 监听播放状态
    [self.playerItem addObserver:self forKeyPath:@"status" options:NSKeyValueObservingOptionNew context:nil];
    // 监听缓冲进度
    [self.playerItem addObserver:self forKeyPath:@"loadedTimeRanges" options:NSKeyValueObservingOptionNew context:nil];
    [self.playerItem removeObserver:self forKeyPath:@"playbackBufferEmpty" context:nil];

Apple 为我们提供了三种播放状态:

  • AVPlayerStatusReadyToPlay : 说明已经准备ok,可以播放了;
  • AVPlayerStatusFailed: 可以通过 playerItem.error 信息来获取失败原因,例如比较常见的错误码 playerItem.error.code == NSURLErrorNotConnectedToInternet || playerItem.error.code == NSURLErrorCannotFindHost ,可以给出找不到网络的错误提示;
  • ** AVPlayerStatusUnknown**: 尝试下载资源但发生未知错误

另外在项目中还会出现缓存不足无法播放的情况,这种情况我们需要添加对 playbackBufferEmpty 的KVO观察, 如果self.playbackBufferEmpty == YES,则视频无法正常播放。
实际情况下可能还有其他状态,比如网络状态的好坏转换会影响视频的播放,对于不同的状态我们都应该有相应的处理,以保证视频的正常播放。

一般的视频播放器都有下载进度条的显示,通过 KVO 方法监听 loadedTimeRanges 属性我们可以得到视频缓冲的进度,具体实现可以查看下面代码。

typedef NS_ENUM(NSInteger, AVPlayerStatus) {
    AVPlayerStatusUnknown,
    AVPlayerStatusReadyToPlay,
    AVPlayerStatusFailed
};

// KVO 方法
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSKeyValueChangeKey,id> *)change context:(void *)context {
    AVPlayerItem *playerItem = (AVPlayerItem *)object;
    if ([keyPath isEqualToString:@"status"]) {
        AVPlayerStatus status = [[change objectForKey:@"new"] integerValue];
        if (status == AVPlayerStatusReadyToPlay) {
            // 停止缓存动画,开始播放
            // 设置视频的总时长
            if ([self.delegate respondsToSelector:@selector(videoTotalTime:)]) {
                [self.delegate videoTotalTime:CMTimeGetSeconds(self.player.currentItem.duration)];
            }
            
        } else if (status == AVPlayerStatusFailed) {
            NSLog(@"AVPlayerStatusFailed == %@", playerItem.error);
            return;
        } else if (status == AVPlayerStatusUnknown) {
            return;
        }
    } else if ([keyPath isEqualToString:@"loadedTimeRanges"]) {
        // 处理缓冲进度条
        NSTimeInterval bufferTime = [self currentVideoLoadedTime];
        NSTimeInterval totalTime = CMTimeGetSeconds(self.player.currentItem.duration);
        CGFloat progress = bufferTime/totalTime;
        if ([self.delegate respondsToSelector:@selector(videoPlayer: loadProgress:)]) {
            [self.delegate videoPlayer:self loadProgress:progress];
        }
    } else if ([keyPath isEqualToString:@"playbackBufferEmpty"]) {
    
    } 
}

关于播放的进度需要处理,重点是addPeriodicTimeObserverForInterval:queue:usingBlock: 方法,该方法返回当前播放的 timeline,当视频暂停、播放或者跳到某一时间进行播放的时候都会调用该方法。需要注意的是暂停和重新播放并不需要我们做额外的操作,只需要调用 pause 或者 play 方法即可,时间的问题该方法已经帮我们处理好了。每次调用该方法对应的要调用 -removeTimeObserver: 对其进行移除,避免发生未知的错误。

 __weak typeof(self) weakSelf = self;
    self.playbackTimeObserver = [self.player addPeriodicTimeObserverForInterval:CMTimeMake(1, 30) queue:NULL usingBlock:^(CMTime time) {
        // 播放进度条以及时间的显示
        if ([weakSelf.delegate respondsToSelector:@selector(videoCurrentTime:)]) {
            Float64 durationTime = CMTimeGetSeconds(weakSelf.playerItem.currentTime);
            [weakSelf.delegate videoCurrentTime:durationTime];
        } 
    }];

// 移除操作
 [self.player removeTimeObserver:_playbackTimeObserver];

关于跳转到某一时刻进行播放,只需要执行下面的方法即可,为了跳转到正确的位置,需要将 toleranceBefore 和 toleranceAfter 都设置为 kCMTimeZero。

- (void)seekToTime:(CMTime)time toleranceBefore:(CMTime)toleranceBefore toleranceAfter:(CMTime)toleranceAfter;

到此位置播放逻辑已经基本实现,调用 [self.player play]; 即可实现视频的播放。

在具体实现过程中我们需要添加视频播放结束通知,以便做下一步处理

   // 添加视频播放结束通知
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(playDidEndNotification:) name:AVPlayerItemDidPlayToEndTimeNotification object:nil];

// 这里设置视频播放结束就自动重播,也可以停止播放显示重播按钮,具体处理看需求
- (void)playDidEndNotification:(NSNotification *)notification {
    self.playEnd = YES;
    [self.delegate isVideoEnd:self.playEnd];
    // 自动重播
    [self.player seekToTime:CMTimeMakeWithSeconds(0, 600) toleranceBefore:kCMTimeZero toleranceAfter:kCMTimeZero];
    [self.player play];
}

播放器样式实现

对于进度条,播放暂停按钮等 UI 的实现在另一个类中进行处理,两者可以通过 protocol 来进行数据交互处理,具体的 UI 代码这里就不一一列出了,可以到 github 上下载简单的demo 来看一下 。

19B87A1D-D2AA-4012-9225-28A5E71AC1E0.png

播放逻辑与 UI 的展示通过 HJPlayView 进行组合,可以通过 HJPlayViewType 来选择不同的 UI 样式,demo 中只给出了全屏播放的 UI 样子,列表页的可以通过继承 HJMaskView 来自己实现,只需要在以下方法进行添加就好。

- (void)setType:(HJPlayViewType)type {
    - (void)setType:(HJPlayViewType)type {
    if (type == HJPlayViewTypeForPlay) {
        _maskView = [self maskView];
    }
    if (type == HJPlayViewTypeForScan) {
        //添加不同的样式即可
    }
}
}

这里只是实现了最简单的播放效果,继续学习~~~

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

推荐阅读更多精彩内容