好消息,音视频播放器重新大改版2.0~
动态切换内核,支持边下边播的播放器方案
- 支持音/视频播放,midi文件播放
- 支持在线播放/本地播放
- 支持后台播放,音频提取播放
- 支持视频边下边播,分片下载播放存储
- 支持断点续载续播,下次直接优先从缓冲读取播放
- 支持缓存管理,清除时间段缓存
- 支持试看,自动跳过片头
- 支持记录上次播放时间
- 支持自动播放,自动连续播放
- 支持随机/重复/顺序播放
- 支持重力感应,全屏/半屏切换
- 支持基本手势操作,进度音量等
- 支持切换不同分辨率视频
- 支持直播流媒体播放
- 持续更新ing...
视频支持格式:mp4、m3u8、wav、avi
音频支持格式:midi、mp3、
pod 'KJPlayer' # 播放器功能区
pod 'KJPlayer/AVPlayer' # AVPlayer内核播放器
pod 'KJPlayer/AVDownloader' # AVPlayer附加边播边下边存分支
pod 'KJPlayer/MIDI' # midi内核
pod 'KJPlayer/IJKPlayer' # ijkplayer内核
横屏展示效果图:
竖屏目前展示效果:
所有播放器壳子都是基于该基础做处理,提取公共部分
API & Property | 类型 | 功能 |
---|---|---|
delegate | Property | 委托代理 |
requestHeader | Property | 视频请求头 |
roregroundResume | Property | 返回前台继续播放 |
backgroundPause | Property | 进入后台暂停播放 |
autoPlay | Property | 是否开启自动播放 |
speed | Property | 播放速度 |
volume | Property | 播放音量 |
cacheTime | Property | 缓存达到多少秒才能播放 |
skipHeadTime | Property | 跳过片头 |
timeSpace | Property | 时间刻度 |
kVideoTotalTime | Property | 获取视频总时长 |
kVideoURLFromat | Property | 获取视频格式 |
kVideoTryLookTime | Property | 免费试看时间和试看结束回调 |
videoURL | Property | 视频地址 |
localityData | Property | 是否为本地资源 |
isPlaying | Property | 是否正在播放 |
currentTime | Property | 当前播放时间 |
ecode | Property | 播放失败 |
kVideoAdvanceAndReverse | Property | 快进或快退 |
shared | Property | 单例属性 |
kj_sharedInstance | Instance | 创建单例 |
kj_attempDealloc | Instance | 销毁单例 |
kj_play | Instance | 准备播放 |
kj_replay | Instance | 重播 |
kj_resume | Instance | 继续 |
kj_pause | Instance | 暂停 |
kj_stop | Instance | 停止 |
/* 当前播放器状态 */
- (void)kj_player:(KJBasePlayer*)player state:(KJPlayerState)state;
/* 播放进度 */
- (void)kj_player:(KJBasePlayer*)player currentTime:(NSTimeInterval)time;
/* 缓存进度 */
- (void)kj_player:(KJBasePlayer*)player loadProgress:(CGFloat)progress;
/* 播放错误 */
- (void)kj_player:(KJBasePlayer*)player playFailed:(NSError*)failed;
播放器UI相关协议
API & Property | 类型 | 功能 |
---|---|---|
playerView | Property | 播放器载体 |
background | Property | 背景颜色 |
placeholder | Property | 占位图 |
videoGravity | Property | 视频显示模式 |
kVideoSize | Property | 获取视频尺寸大小 |
kVideoTimeScreenshots | Property | 获取当前截屏 |
只要子控件没有涉及到手势交互,我均采用Layer的方式来处理,然后根据zPosition
来区分控件的上下层级关系
/* 委托代理 */
@property (nonatomic,weak) id <KJPlayerBaseViewDelegate> delegate;
/* 主色调,默认白色 */
@property (nonatomic,strong) UIColor *mainColor;
/* 副色调,默认红色 */
@property (nonatomic,strong) UIColor *viceColor;
/* 支持手势,支持多枚举 */
@property (nonatomic,assign) KJPlayerGestureType gestureType;
/* 长按执行时间,默认1秒 */
@property (nonatomic,assign) NSTimeInterval longPressTime;
/* 操作面板自动隐藏时间,默认2秒然后为零表示不隐藏 */
@property (nonatomic,assign) NSTimeInterval autoHideTime;
/* 操作面板高度,默认60px */
@property (nonatomic,assign) CGFloat operationViewHeight;
/* 当前操作面板状态 */
@property (nonatomic,assign,readonly) BOOL displayOperation;
/* 隐藏操作面板时是否隐藏返回按钮,默认yes */
@property (nonatomic,assign) BOOL isHiddenBackButton;
/* 小屏状态下是否显示返回按钮,默认yes */
@property (nonatomic,assign) BOOL smallScreenHiddenBackButton;
/* 全屏状态下是否显示返回按钮,默认no */
@property (nonatomic,assign) BOOL fullScreenHiddenBackButton;
/* 是否为全屏,名字别乱改后面kvc有使用 */
@property (nonatomic,assign) BOOL isFullScreen;
/* 当前屏幕状态,名字别乱改后面kvc有使用 */
@property (nonatomic,assign,readonly) KJPlayerVideoScreenState screenState;
/* 当前屏幕状态发生改变 */
@property (nonatomic,copy,readwrite) void (^kVideoChangeScreenState)(KJPlayerVideoScreenState state);
/* 返回回调 */
@property (nonatomic,copy,readwrite) void (^kVideoClickButtonBack)(KJBasePlayerView *view);
/* 提示文字面板属性,默认最大宽度250px */
@property (nonatomic,copy,readonly) void (^kVideoHintTextInfo)(void(^)(KJPlayerHintInfo *info));
#pragma mark - 控件
/* 快进快退进度控件 */
@property (nonatomic,strong) KJPlayerFastLayer *fastLayer;
/* 音量亮度控件 */
@property (nonatomic,strong) KJPlayerSystemLayer *vbLayer;
/* 加载动画层 */
@property (nonatomic,strong) KJPlayerLoadingLayer *loadingLayer;
/* 文本提示框 */
@property (nonatomic,strong) KJPlayerHintTextLayer *hintTextLayer;
/* 顶部操作面板 */
@property (nonatomic,strong) KJPlayerOperationView *topView;
/* 底部操作面板 */
@property (nonatomic,strong) KJPlayerOperationView *bottomView;
#pragma mark - method
/* 隐藏操作面板,是否隐藏返回按钮 */
- (void)kj_hiddenOperationView;
/* 显示操作面板 */
- (void)kj_displayOperationView;
/* 单双击手势反馈 */
- (void)kj_basePlayerView:(KJBasePlayerView*)view isSingleTap:(BOOL)tap;
/* 长按手势反馈 */
- (void)kj_basePlayerView:(KJBasePlayerView*)view longPress:(UILongPressGestureRecognizer*)longPress;
/* 进度手势反馈,不替换UI请返回当前时间和总时间,范围-1 ~ 1 */
- (NSArray*)kj_basePlayerView:(KJBasePlayerView*)view progress:(float)progress end:(BOOL)end;
/* 音量手势反馈,是否替换自带UI,范围0 ~ 1 */
- (BOOL)kj_basePlayerView:(KJBasePlayerView*)view volumeValue:(float)value;
/* 亮度手势反馈,是否替换自带UI,范围0 ~ 1 */
- (BOOL)kj_basePlayerView:(KJBasePlayerView*)view brightnessValue:(float)value;
枚举文件夹和公共方法管理
- KJPlayerState:播放器状态
- KJPlayerCustomCode:错误code
- KJPlayerGestureType:手势操作
- KJPlayerPlayType:播放类型
- KJPlayerDeviceDirection:手机方向
- KJPlayerVideoGravity:播放器充满类型
- KJPlayerVideoFromat:视频格式
主要包括两部分,数据库模型和增删改查等工具
数据库结构
dbid:唯一id,视频链接去除scheme然后md5
videoUrl:视频链接
saveTime:存储时间戳
sandboxPath:沙盒地址
videoFormat:视频格式
videoTime:视频时间
videoData:视频数据
数据库工具
方法 | 功能 |
---|---|
kj_insertData:Data: | 插入数据,重复数据替换处理 |
kj_deleteData: | 删除数据 |
kj_addData: | 新添加数据 |
kj_updateData:Data: | 更新数据 |
kj_checkData: | 查询数据,传空传全部数据 |
kCheckAppointDatas | 指定条件查询 |
中间桥梁作用,把网络请求缓存到本地的临时数据传递给播放器
工作流程:
1、获取视频类型,根据网址来确定,目前没找到更好的方式(知道的朋友可以指点一下)
/// 根据链接获取Asset类型
NS_INLINE KJPlayerAssetType kPlayerVideoAesstType(NSURL *url){
if (url == nil) return KJPlayerAssetTypeNONE;
if (url.pathExtension.length) {
if ([url.pathExtension containsString:@"m3u8"] || [url.pathExtension containsString:@"ts"]) {
return KJPlayerAssetTypeHLS;
}
}
NSArray * array = [url.path componentsSeparatedByString:@"."];
if (array.count == 0) {
return KJPlayerAssetTypeNONE;
}else{
if ([array.lastObject containsString:@"m3u8"] || [array.lastObject containsString:@"ts"]) {
return KJPlayerAssetTypeHLS;
}
}
return KJPlayerAssetTypeFILE;
}
2、处理视频,这里才用队列组来处理,子线程处理解决第一次加载卡顿问题
dispatch_group_async(group, dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
if ([weakself kj_dealVideoURL:&tempURL]) {
if (![tempURL.absoluteString isEqualToString:self->_videoURL.absoluteString]) {
self->_videoURL = tempURL;
[weakself kj_initPreparePlayer];
}else{
[weakself kj_playerReplay];
}
}
});
3、处理视频链接地址,这里分两种情况,使用缓存就从缓存当中读取
/* 判断当前资源文件是否有缓存,修改为指定链接地址 */
- (void)kj_judgeHaveCacheWithVideoURL:(NSURL * _Nonnull __strong * _Nonnull)videoURL{
self.locality = NO;
KJCacheManager.kJudgeHaveCacheURL(^(BOOL locality) {
self.locality = locality;
if (locality) {
self.playError = [DBPlayerDataInfo kj_errorSummarizing:KJPlayerCustomCodeCachedComplete];
}
}, videoURL);
}
获取数据库当中的数据
/* 判断是否有缓存,返回缓存链接 */
+ (void(^)(void(^)(BOOL),NSURL * _Nonnull __strong * _Nonnull))kJudgeHaveCacheURL{
return ^(void(^locality)(BOOL),NSURL * _Nonnull __strong * _Nonnull videoURL){
NSArray<DBPlayerData*>*temps = [DBPlayerDataInfo kj_checkData:kPlayerIntactName(*videoURL)];
BOOL boo = NO;
if (temps.count) {
DBPlayerData *data = temps.firstObject;
NSString *path = data.sandboxPath;
if (data.videoIntact && [KJCacheManager kj_haveFileSandboxPath:&path]) {
//移出之前的临时文件
NSString *tempPath = [path stringByAppendingPathExtension:kTempReadName];
[[NSFileManager defaultManager] removeItemAtPath:tempPath error:NULL];
*videoURL = [NSURL fileURLWithPath:path];
boo = YES;
}
}
kGCD_player_main(^{
if (locality) locality(boo);
});
};
}
4、判断地址是否可用,添加下载和播放桥梁
PLAYER_WEAKSELF;
if (!kPlayerHaveTracks(*videoURL, ^(AVURLAsset * asset) {
if (weakself.useCacheFunction && !weakself.localityData) {
weakself.state = KJPlayerStateBuffering;
weakself.loadState = KJPlayerLoadStateNone;
NSURL * tempURL = weakself.connection.kj_createSchemeURL(*videoURL);
weakself.asset = [AVURLAsset URLAssetWithURL:tempURL options:weakself.requestHeader];
[weakself.asset.resourceLoader setDelegate:weakself.connection queue:dispatch_get_main_queue()];
}else{
weakself.asset = asset;
}
}, self.requestHeader)) {
self.ecode = KJPlayerCustomCodeVideoURLFault;
self.state = KJPlayerStateFailed;
[self kj_destroyPlayer];
return NO;
}
5、播放准备操作设置playerItem
,然后初始化player
,添加时间观察者处理播放
self.timeObserver = [self.player addPeriodicTimeObserverForInterval:CMTimeMakeWithSeconds(_timeSpace, NSEC_PER_SEC) queue:dispatch_queue_create("kj.player.time.queue", NULL) usingBlock:^(CMTime time) {
NSTimeInterval sec = CMTimeGetSeconds(time);
if (isnan(sec) || sec < 0) sec = 0;
if (weakself.totalTime <= 0) return;
if ((NSInteger)sec >= (NSInteger)weakself.totalTime) {
[weakself.player pause];
weakself.state = KJPlayerStatePlayFinished;
weakself.currentTime = 0;
}else if (weakself.userPause == NO && weakself.buffered) {
weakself.state = KJPlayerStatePlaying;
weakself.currentTime = sec;
}
if (sec > weakself.tryTime && weakself.tryTime) {
[weakself kj_pause];
if (!weakself.tryLooked) {
weakself.tryLooked = YES;
kGCD_player_main(^{
if (weakself.tryTimeBlock) weakself.tryTimeBlock();
});
}
}else{
weakself.tryLooked = NO;
}
}];
6、处理视频状态,kvo监听播放器五种状态
status
:监听播放器状态loadedTimeRanges
:监听播放器缓冲进度presentationSize
:监听视频尺寸playbackBufferEmpty
:监听缓存不够的情况playbackLikelyToKeepUp
:监听缓存足够
大致流程就差不多这样子,Demo也写的很详细,可以自己去看看
也可以通过以下方式联系我,邮箱地址:ykj310@126.com