iOS 基于AVPLayer封装视频播放器

最近的公司有个需求,需要做客户端播放远程视频。本来需求很简单,只要能播放、暂停、拖动进度就行啦。
原定技术方案使用系统自带的播放controller,结果经过调研发现系统播放器controller 有很多问题,原因如下:

注意:apple 原来提供的播放类
 视频播放(播放基类)

 1> AVPlayer
 能播放本地、远程的音频、视频文件
 基于Layer显示,得自己去编写控制面板
 
 2> MPMoviePlayerController 
 能播放本地、远程的音频、视频文件
 自带播放控制面板(暂停、播放、播放进度、是否要全屏)
注意:关于MPMoviePlayerController 我们看一下官方文档是这样说的:The MPMoviePlayerController class is formally deprecated in iOS 9
 
 3> AVPlayerViewController 
 能播放本地、远程的音频、视频文件
 内部是封装了MPMoviePlayerController
 播放界面默认就是全屏的
 如果播放功能比较简单,仅仅是简单地播放远程、本地的视频文件,建议用这个
 
注意:关于AVPlayerViewController 我们看一下官方文档是这样说的:Available in iOS 8.0 and later

完蛋了,梦想破灭了,本来想一劳永逸,结果:
MPMoviePlayerController 在iOS9以后被弃用,
AVPlayerViewController 只能用于iOS8以后。我们的版本必须要支持iOS7以后。

好吧那就自己动手封装一个播放器吧!

先看看基本的模块架构设计吧:

基本设计图

OK!设计好了开始动手coding。

先定义一个枚举:

typedef NS_ENUM(NSInteger,AVPlayerPlayState) {


    AVPlayerPlayStatePreparing = 0x0, // 准备播放
    AVPlayerPlayStateBeigin,       // 开始播放
    AVPlayerPlayStatePlaying,      // 正在播放
    AVPlayerPlayStatePause,        // 播放暂停
    AVPlayerPlayStateEnd,          // 播放结束
    AVPlayerPlayStateBufferEmpty,  // 没有缓存的数据供播放了
    AVPlayerPlayStateBufferToKeepUp,//有缓存的数据可以供播放
    
    AVPlayerPlayStateNotPlay,      // 不能播放
    AVPlayerPlayStateNotKnow       // 未知情况
};

改枚举指明了各种播放状态。

Manager 的实现是这样的,看一下.h文件:

@interface MFPlayerPlayVideoManager : NSObject

@property (nonatomic, weak) id<PlayVideoDelegate> delegate;
@property (nonatomic, assign, readonly) BOOL isPlaying;

+ (instancetype)sharedInstance;

- (void)playVideoFromhUrl:(NSURL *)url onView:(UIView *)playView;

- (CMTime)playerCurrentDuration;
- (CMTime)playerItemDuration;

- (void)updateMovieScrubberControl:(CGFloat)time;
- (void)setVideoFillMode:(NSString *)fillMode;

// 开始拖动
- (void)beiginSliderScrubbing;
// 结束拖动
- (void)endSliderScrubbing:(CGFloat)tolerance;
// 拖动值发生改变
- (void)sliderScrubbing:(CGFloat)time;

// 播放控制
- (void)play;
- (void)pause;
- (void)resum;
- (void)stop;
- (void)clear;

@end

是的,这个Manager 是最核心的内,定义的方法也很简单,不用说相信你也看得懂。

Manager 的业务发生变化需要对外通知 UIView 的显示掉的时候,就调用这个代理:
@property (nonatomic, weak) id<PlayVideoDelegate> delegate;

这样将业务逻辑和UI完全分开,PlayVideoManager 里面全是播放业务,MFPlayerPlayVideoView 里面全是UI显示相关联的。

当UI的改变需要控制业务的变化时,就调用Manager 对外暴露的方法:

// 开始拖动
- (void)beiginSliderScrubbing;
// 结束拖动
- (void)endSliderScrubbing:(CGFloat)tolerance;
// 拖动值发生改变
- (void)sliderScrubbing:(CGFloat)time;

// 播放控制
- (void)play;
- (void)pause;
- (void)resum;
- (void)stop;
- (void)clear;

MFPlayerPlayVideoManager.h 中有个比较蛋痛的东西,那就是是监听是否好友缓冲数据可以播放,理论上实现以下监听就可以啦:

    [self.mPlayerItem  addObserver:self
                        forKeyPath:@"playbackBufferEmpty"
                           options:NSKeyValueObservingOptionNew
                           context:kPlaybackBufferEmptyObservationContext];
    
    [self.mPlayerItem  addObserver:self
                        forKeyPath:@"playbackLikelyToKeepUp"
                           options:NSKeyValueObservingOptionNew
                           context:kPlaybackLikelyToKeepUpObservationContext];

这样当播放的缓冲数据没有的时候 kPlaybackBufferEmptyObservationContext 监听就会被回调,当网络OK又有缓冲数据的时候 kPlaybackLikelyToKeepUpObservationContext 监听就会被回调,能监听到这两个状态,我们就可以做一些自由发挥的操作了。比如没有冲昏的时候我们可以暂停,并给出用户友好提示,如果再次有缓存数据可以继续播放并界面做一些调整。

很可惜,经过本人实验证明:kPlaybackBufferEmptyObservationContextkPlaybackBufferEmptyObservationContext监听的调用时机相当的混乱,没有缓存时kPlaybackBufferEmptyObservationContext会被回调,用户手动点击暂停方法也会被回调。kPlaybackBufferEmptyObservationContext再次获得缓冲数据会被回调,用户点击开始播放也会被回调。还有一些情况也会被回调。如果大家感兴趣可以自己打断点跟踪一些他们的回调

- (void)observeValueForKeyPath:(NSString*) path
                      ofObject:(id)object
                        change:(NSDictionary*)change
                       context:(void*)context

所以,我在此将缓冲状态的监听放到了rate 里面:

       /* 监听 AVPlayer "rate" 属性 以便我们去更新播放进度控件. */
        [self.mPlayer addObserver:self
                      forKeyPath:@"rate"
                         options:NSKeyValueObservingOptionInitial | NSKeyValueObservingOptionNew
                         context:kRateObservationContext];

在监听方法 observeValueForKeyPath:ofObject:change:context:实现四这样的:

if (context == kRateObservationContext){

        /**
         *  暂停分两种:一个强制暂停(以就是点击了暂停按钮)
         *  另一种就是网络不好加载卡住了暂停。
         */
        if (self.mPlayer.rate == 0) {
            
            /* 缓存不够导致的暂停 */
            if (!_isForcusPause) {
                NSLog(@"self.mPlayer.rate == 0 && _isEmptyBuffer---AVPlayerPlayStatePreparing");
                [self updateCurrentPlayStatus:AVPlayerPlayStatePreparing];
                _isEmptyBufferPause = YES;
            }
            /* 正常情况下导致的暂停 */
            else{
                [self updateCurrentPlayStatus:AVPlayerPlayStatePause];
            }

        }
        /**
         *  播放都一样
         */
        if (self.mPlayer.rate == 1) {
            _isForcusPause = NO;
            _isEmptyBufferPause = NO;
            NSLog(@"self.mPlayer.rate == 1----AVPlayerPlayStatePreparing");
            [self updateCurrentPlayStatus:AVPlayerPlayStateBeigin];
        }

    }

注意:self.mPlayer.rate == 0 标示目前是暂停状态,self.mPlayer.rate == 1 标示目前是播放状态。不管是什么原因导致的暂停和播放,kRateObservationContext监听都会被调用。isForcusPause == YES 标示的是强制暂停,是用户行为导致的。否则就是网络问题导致的暂停。

看看运行效果:

Paste_Image.png

Github源码
https://github.com/wmf00032/Resource

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

推荐阅读更多精彩内容