iOS已经提供了一些视频播放的API,封装性很强,使用上也比较方便,但是自定制UI比较困难,自由度比较低:
- MPMoviePlayerController
- AVPictureInPictureController
所以我们自己用AVPlayer实现一些视频播放的定制开发。AVPlayer属于AVFoundation框架,可以播放视频、音频,支持本地、网络视频源,相对来说更加接近底层。
实现视频播放
AVFoundtion 框架中主要使用 AVAsset 类来展示媒体信息,了解几个常用类:
1、AVAsset:AVAsset类专门用于获取多媒体的相关信息,包括获取多媒体的画面、声音等信息,属于一个抽象类,不能直接使用。
2、AVURLAsset:AVAsset的子类,可以根据一个URL路径创建一个包含媒体信息的AVURLAsset对象。
3、AVPlayerItem:一个媒体资源管理对象,管理者视频的一些基本信息和状态,一个AVPlayerItem对应着一个视频资源。
在AVPlayer实现视频播放的开发中,AVPlayerLayer 显示视频,AVPlayerItem 提供视频信息, AVPlayer 管理和调控.
AVPlayer 本身并不能显示视频, 显示视频的是 AVPlayerLayer。 AVPlayerLayer 继承自 CALayer。
初始化操作:
AVPlayer初始化
- (AVPlayer *)avPlayer
{
if (_avPlayer == nil) {
_avPlayer = [[AVPlayer alloc] init];
// 设置默认音量
_avPlayer.volume = 0.5;
}
return _avPlayer;
}
- (instancetype)init
{
if (self = [super init]) {
// 让view的layerClass为AVPlayerLayer类,那么self.layer就为AVPlayerLayer的实例
self.playerLayer = (AVPlayerLayer *)self.layer;
self.playerLayer.backgroundColor = [[UIColor blackColor] colorWithAlphaComponent:0.6].CGColor;
// 初始化playerLayer的player
self.playerLayer.player = self.avPlayer;
}
return self;
}
设置媒体资源
- (void)settingPlayerItem:(AVPlayerItem *)playerItem
{
_playerItem = playerItem;
[self.avPlayer pause];
/*
replaceCurrentItemWithPlayerItem: 用于切换视频
*/
// 设置当前playerItem
[self.avPlayer replaceCurrentItemWithPlayerItem:playerItem];
[self.avPlayer play];
}
视频播放
- (void)viewDidLoad {
[super viewDidLoad];
CGFloat palyerW = [UIScreen mainScreen].bounds.size.width;
NKAVPlayerView *playerView = [[NKAVPlayerView alloc] init];
playerView.frame = CGRectMake(0, 0, palyerW, palyerW / 7 * 4);
[self.view addSubview:playerView];
NSString *moviePath = [[NSBundle mainBundle] pathForResource:@"movie" ofType:@"mp4"];
[playerView settingPlayerItemWithUrl:[NSURL fileURLWithPath:moviePath]];
}
这样视频播放就完成,附上运行效果:
视频播放的控制
- 监控播放状态
// 监控它的status也可以获得播放状态
[self.avPlayer.currentItem addObserver:self forKeyPath:@"status" options:NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew context:nil];
- 监控缓冲加载
//监控缓冲加载
[self.avPlayer.currentItem addObserver:self forKeyPath:@"loadedTimeRanges" options:NSKeyValueObservingOptionOld | NSKeyValueObservingOptionNew context:nil];
- 监控播放完成
//监控播放完成
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(playFinished:) name:AVPlayerItemDidPlayToEndTimeNotification object:self.avPlayer.currentItem];
- 监控时间进度
//监控时间进度(根据API提示,如果要监控时间进度,这个对象引用计数器要+1,retain)
__weak typeof(self) weakSelf = self;
self.timeObserver = [self.avPlayer addPeriodicTimeObserverForInterval:CMTimeMake(1, 1) queue:dispatch_get_main_queue() usingBlock:^(CMTime time) {
// 获取 item 当前播放秒
float currentPlayTime = (double)weakSelf.avPlayer.currentItem.currentTime.value/ weakSelf.avPlayer.currentItem.currentTime.timescale;
[weakSelf updateVideoSlider:currentPlayTime];
}];
- 横竖屏切换
// 添加通知
[[UIDevice currentDevice] beginGeneratingDeviceOrientationNotifications];
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(orientationHandler)
name:UIDeviceOrientationDidChangeNotification
object:nil
];
// 切换到大屏
- (void)fullScreenButtonClick {
[self forceChangeOrientation:UIInterfaceOrientationLandscapeRight];
self.controlView.fullScreenButton.hidden = YES;
self.controlView.shrinkScreenButton.hidden = NO;
self.frame = CGRectMake(0, 0, [UIScreen mainScreen].bounds.size.width, [UIScreen mainScreen].bounds.size.height);
}
// 切换到小屏
- (void)shrinkScreenButtonClick {
[self forceChangeOrientation:UIInterfaceOrientationPortrait];
self.controlView.fullScreenButton.hidden = NO;
self.controlView.shrinkScreenButton.hidden = YES;
self.frame = _shrinkRect;
}
/**
* 强制横屏
*
* @param orientation 横屏方向
*/
- (void)forceChangeOrientation:(UIInterfaceOrientation)orientation
{
int val = orientation;
if ([[UIDevice currentDevice] respondsToSelector:@selector(setOrientation:)]) {
SEL selector = NSSelectorFromString(@"setOrientation:");
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:[UIDevice instanceMethodSignatureForSelector:selector]];
[invocation setSelector:selector];
[invocation setTarget:[UIDevice currentDevice]];
[invocation setArgument:&val atIndex:2];
[invocation invoke];
}
}
AVQueuePlayer
AVQueuePlayer集成于AVPlayer,主要实现添加多个AVPlayerItem,然后顺序播放。
总结一下播放视频的步骤
首先,得到视频的URL
根据URL创建AVPlayerItem
把AVPlayerItem 提供给 AVPlayer
AVPlayerLayer 显示视频。
AVPlayer 控制视频, 播放, 暂停, 跳转 等等。
播放过程中获取缓冲进度,获取播放进度。
视频播放完成后做些什么,是暂停还是循环播放,还是获取最后一帧图像。
后续继续更新一些小问题,进度条、弹出动画等等
附上demo,如果帮到你了请给个小星星
GitHub