AVPlayer的基本使用

在iOS開發中,播放視頻一般有兩種方式,一種是使用MPMoviePlayerController(須要導入MediaPlayer.Framework),還有一種是使用AVPlayer。關於這兩個類的區別能夠參考http://stackoverflow.com/questions/8146942/avplayer-and-mpmovieplayercontroller-differences,簡而言之就是MPMoviePlayerController使用更簡單,功能不如AVPlayer強大,而AVPlayer使用稍微麻煩點,不過功能更增強大。這篇博客主要介紹下AVPlayer的基本使用,因爲博主也是剛剛接觸,因此有問題你們直接指出~ html

  在開發中,單純使用AVPlayer類是沒法顯示視頻的,要將視頻層添加至AVPlayerLayer中,這樣才能將視頻顯示出來,因此先在ViewController的@interface中添加如下屬性 git

@property (nonatomic ,strong) AVPlayer *player; @property (nonatomic ,strong) AVPlayerItem *playerItem;
@property (nonatomic ,weak) IBOutletPlayerView *playerView;

其中playerView繼承自UIView,不太重寫了set和get方法,用於將player添加至playerView的AVPlayerLayer中,這樣才能順利將視頻顯示出來 github

在PlayerView.h中聲明一個AVPlayer對象,因爲默認的layer是CALayer,而AVPlayer只能添加至AVPlayerLayer中,因此咱們改變一下layerClass,這樣PlayerView的默認layer就變了,以後咱們能夠把在viewController中初始化的AVPlayer對象賦給AVPlayerLayer的player屬性。 app

複製代碼

 PlayerView.h  ide

 @property (nonatomic ,strong) AVPlayer *player; atom

 

 PlayerView.m spa

+ (Class)layerClass { return [AVPlayerLayer class]; } - (AVPlayer *)player { return [(AVPlayerLayer *)[self layer] player]; } - (void)setPlayer:(AVPlayer *)player { [(AVPlayerLayer *)[self layer] setPlayer:player]; }
複製代碼

而後在viewDidLoad中執行初始化: 線程

複製代碼
NSURL *videoUrl = [NSURL URLWithString:@"http://www.jxvdy.com/file/upload/201405/05/18-24-58-42-627.mp4"]; self.playerItem = [AVPlayerItem playerItemWithURL:videoUrl]; [self.playerItem addObserver:self forKeyPath:@"status" options:NSKeyValueObservingOptionNew context:nil];// 監聽status屬性 [self.playerItem addObserver:self forKeyPath:@"loadedTimeRanges" options:NSKeyValueObservingOptionNew context:nil];// 監聽loadedTimeRanges屬性 self.player = [AVPlayer playerWithPlayerItem:self.playerItem]; [[NSNotificationCenterdefaultCenter]addObserver:selfselector:@selector(moviePlayDidEnd:) name:AVPlayerItemDidPlayToEndTimeNotificationobject:self.playerItem];
複製代碼

先將在線視頻連接存放在videoUrl中,而後初始化playerItem,playerItem是管理資源的對象(A player item manages the presentation state of an asset with which it is associated. A player item contains player item tracks—instances of AVPlayerItemTrack—that correspond to the tracks in the asset.) code

對象的關係

而後監聽playerItem的status和loadedTimeRange屬性,status有三種狀態: orm

AVPlayerStatusUnknown,

AVPlayerStatusReadyToPlay,

AVPlayerStatusFailed

當status等於AVPlayerStatusReadyToPlay時表明視頻已經能夠播放了,咱們就能夠調用play方法播放了。

loadedTimeRange屬性表明已經緩衝的進度,監聽此屬性能夠在UI中更新緩衝進度,也是頗有用的一個屬性。

最後添加一個通知,用於監聽視頻是否已經播放完畢,而後實現KVO的方法:

複製代碼
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context { AVPlayerItem *playerItem = (AVPlayerItem *)object; if ([keyPath isEqualToString:@"status"]) { if ([playerItem status] == AVPlayerStatusReadyToPlay) { NSLog(@"AVPlayerStatusReadyToPlay"); self.stateButton.enabled = YES; CMTime duration = self.playerItem.duration;// 獲取視頻總長度 CGFloat totalSecond = playerItem.duration.value / playerItem.duration.timescale;// 轉換成秒 _totalTime = [self convertTime:totalSecond];// 轉換成播放時間 [self customVideoSlider:duration];// 自定義UISlider外觀 NSLog(@"movie total duration:%f",CMTimeGetSeconds(duration)); [self monitoringPlayback:self.playerItem];// 監聽播放狀態 } else if ([playerItem status] == AVPlayerStatusFailed) { NSLog(@"AVPlayerStatusFailed"); } } else if ([keyPath isEqualToString:@"loadedTimeRanges"]) { NSTimeInterval timeInterval = [self availableDuration];// 計算緩衝進度 NSLog(@"Time Interval:%f",timeInterval); CMTime duration = self.playerItem.duration; CGFloat totalDuration = CMTimeGetSeconds(duration); [self.videoProgress setProgress:timeInterval / totalDuration animated:YES]; } } - (NSTimeInterval)availableDuration { NSArray *loadedTimeRanges = [[self.playerView.player currentItem] loadedTimeRanges]; CMTimeRange timeRange = [loadedTimeRanges.firstObject CMTimeRangeValue];// 獲取緩衝區域 float startSeconds = CMTimeGetSeconds(timeRange.start); float durationSeconds = CMTimeGetSeconds(timeRange.duration); NSTimeInterval result = startSeconds + durationSeconds;// 計算緩衝總進度 return result; } - (NSString *)convertTime:(CGFloat)second{ NSDate *d = [NSDate dateWithTimeIntervalSince1970:second]; NSDateFormatter *formatter = [[NSDateFormatter alloc] init]; if (second/3600 >= 1) { [formatter setDateFormat:@"HH:mm:ss"]; } else { [formatter setDateFormat:@"mm:ss"]; } NSString *showtimeNew = [formatter stringFromDate:d]; return showtimeNew; }
複製代碼

此方法主要對status和loadedTimeRanges屬性作出響應,status狀態變爲AVPlayerStatusReadyToPlay時,說明視頻已經能夠播放了,這時咱們能夠獲取一些視頻的信息,包含視頻長度等,把播放按鈕設備enabled,點擊就能夠調用play方法播放視頻了。在AVPlayerStatusReadyToPlay的底部還有個monitoringPlayback方法:

複製代碼
- (void)monitoringPlayback:(AVPlayerItem *)playerItem { self.playbackTimeObserver = [self.playerView.player addPeriodicTimeObserverForInterval:CMTimeMake(1, 1) queue:NULL usingBlock:^(CMTime time) { CGFloat currentSecond = playerItem.currentTime.value/playerItem.currentTime.timescale;// 計算當前在第幾秒  [self updateVideoSlider:currentSecond]; NSString *timeString = [self convertTime:currentSecond]; self.timeLabel.text = [NSString stringWithFormat:@"%@/%@",timeString,_totalTime]; }]; }
複製代碼

monitoringPlayback用於監聽每秒的狀態,- (id)addPeriodicTimeObserverForInterval:(CMTime)interval queue:(dispatch_queue_t)queue usingBlock:(void (^)(CMTime time))block;此方法就是關鍵,interval參數爲響應的間隔時間,這裏設爲每秒都響應,queue是隊列,傳NULL表明在主線程執行。能夠更新一個UI,好比進度條的當前時間等。

 

  做爲播放器,除了播放,暫停等功能外。還有一個必不可少的功能,那就是顯示當前播放進度,還有緩衝的區域,個人思路是這樣,用UIProgressView顯示緩衝的可播放區域,用UISlider顯示當前正在播放的進度,固然這裏要對UISlider作一些自定義,代碼以下:

複製代碼
- (void)customVideoSlider:(CMTime)duration { self.videoSlider.maximumValue = CMTimeGetSeconds(duration); UIGraphicsBeginImageContextWithOptions((CGSize){ 1, 1 }, NO, 0.0f); UIImage *transparentImage = UIGraphicsGetImageFromCurrentImageContext(); UIGraphicsEndImageContext(); [self.videoSlider setMinimumTrackImage:transparentImage forState:UIControlStateNormal]; [self.videoSlider setMaximumTrackImage:transparentImage forState:UIControlStateNormal]; }
複製代碼

這樣UISlider就只有中間的ThumbImage了,而ThumbImage左右的顏色都變成透明的了,僅僅是用於顯示當前的播放時間。UIProgressView則用於顯示當前緩衝的區域,不作任何自定義的修改,在StoryBoard看起來是這樣的:

把UISlider添加至UIProgressView上面,運行起來的效果就變成了這樣:

這樣基本的緩衝功能就作好了,固然還有一些功能沒作,好比音量大小,滑動屏幕快進快退等,你們有時間能夠本身作着玩兒下~ 最後的效果以下:

最後附上demo連接:https://github.com/mzds/AVPlayerDemo

相關文章
相關標籤/搜索