目录介绍
- 1.关于此视频封装库介绍
- 1.1 能够满足那些业务需求
- 1.2 对比同类型的库有哪些优势
- 2.关于使用方法说明
- 2.1 关于gradle引用说明
- 2.2 添加布局
- 2.3 最简单的视频播放器参数设定
- 2.4 注意的问题
- 2.5 关于开源库中的类说明
- 3.关于播放类型说明
- 3.1 普通视频播放
- 3.2 list页面视频播放
- 3.3 小窗口视频播放
- 3.4 类似爱奇艺,优酷会员试看视频播放
- 3.5 关于封装库中日志打印
- 4.关于相关方法说明
- 4.1 关于VideoPlayer类[播放器]中方法说明
- 4.2 关于VideoPlayerController类[控制器]中方法说明
- 4.3 关于对象的销毁
- 5.关于封装的思路
- 5.1 参考的案例思路
- 5.2 封装的基本思路
- 5.3 关于窗口切换分析
- 5.4 关于VideoPlayerManager视频播放器管理器分析
- 5.5 关于VideoPlayerController视频控制器分析
- 5.6 关于InterVideoPlayer接口分析
- 6.关于如何自定义你想要的视频播放模式
- 6.1 自定义视频播放器
- 7.关于效果图的展示
- 7.1 效果图如下所示
- 8.关于遇到的问题说明
- 8.1 视频难点
- 8.2 遇到的bug
- 8.3 后期需要实现的功能
- 9.关于版本更新说明
- 9.1 V1.0.0 更新于2017年9月4日
- 9.2 V1.0.1 更新于2017年11月18日
- 9.3 v1.1.0 更新于2018年1月15日
- 10.关于参考文档说明
- 10.1 参考的项目
- 10.2 参考的博客
- 11.关其他说明
- 11.1 目前市场流行的视频框架
- 11.2 如何选择合适的框架
- 11.3 关于我的个人博客和站点
0.备注
- 仿照爱奇艺,优酷播放器写的,十分感谢GitHub上大神前辈们的开源案例和思路。
- 支持插入广告,设置视频观看权限,观看完后登录或者购买会员。我看到在star较多的项目issues中,有些人正好需要这个案例,库集成后直接通过代码调用即可,灵活且拓展性强。
- 由于调到做视频的部门,因此此部分代码会持续更新,也欢迎同行提bug或者问题
- 如果你觉得还可以,给个star吧!我也在持续学习中!!!
- 项目地址:https://github.com/yangchong2...
1.关于此视频封装库介绍
1.1 能够满足那些业务需求
A基础功能
- 1.1.1 能够自定义视频加载loading类型,设置视频标题,设置视频底部图片,设置播放时长等基础功能
- 1.1.2 可以切换播放器的视频播放状态,播放错误,播放未开始,播放开始,播放准备中,正在播放,暂停播放,正在缓冲等等状态
- 1.1.3 可以自由设置播放器的播放模式,比如,正常播放,全屏播放,和小屏幕播放。其中全屏播放支持旋转屏幕。
- 1.1.4 可以支持多种视频播放类型,比如,原生封装视频播放器,还有基于ijkplayer封装的播放器。
- 1.1.5 可以设置是否隐藏播放音量,播放进度,播放亮度等,可以通过拖动seekBar改变视频进度。还支持设置n秒后不操作则隐藏头部和顶部布局功能
-
B高级功能
- 1.1.6 支持一遍播放一遍缓冲的功能,其中缓冲包括两部分,第一种是播放过程中缓冲,第二种是暂停过程中缓冲
- 1.1.7 基于ijkplayer的封装播放器,支持多种格式视频播放
- 1.1.8 可以设置是否记录播放位置,设置播放速度,设置屏幕比例
- 1.1.9 支持滑动改变音量【屏幕右边】,改变屏幕亮度【屏幕左边】,支持切换视频清晰度模式
- 1.1.0 支持list页面中视频播放,滚动后暂停播放,播放可以自由设置是否记录状态。并且还支持删除视频播放位置状态。
-
C拓展功能
- C1产品需求:类似优酷,爱奇艺视频播放器部分逻辑。比如如果用户没有登录也没有看视频权限,则提示试看视频[自定义布局];如果用户没有登录但是有看视频权限,则正常观看;如果用户登录,但是没有充值会员,部分需要权限视频则进入试看模式,试看结束后弹出充值会员界面;如果用户余额不足,比如余额只有99元,但是视频观看要199元,则又有其他提示。
- C2自身需求:比如封装好了视频播放库,那么点击视频上登录按钮则跳到登录页面;点击充值会员页面也跳到充值页面。这个通过定义接口,可以让使用者通过方法调用,灵活处理点击事件。
- C.1.1 实现了上面两个需求,灵活可拓展性强。
- C.1.2 对于设置视频的宽高,建议设置成4:3或者16:9或者常用比例,如果不是常用比例,则可能会有黑边。其中黑边的背景可以设置
- C.1.3 可以设置播放有权限的视频时的各种文字描述,而没有把它写在封装库中,使用者自己设定
- C.1.4 锁定屏幕功能
-
D待添加功能
- D.1.1 可以支持屏幕截图功能,视频添加水印效果
- D.1.2 支持弹幕功能
- D.1.3 后期待定
1.2 对比同类型的库有哪些优势
1.2.1目前仅仅查了下GitHub上项目
- 目前GitHub上比较流行的库
- 至于官方库就不说了,jiecao的库是基于ijkplayer视频框架,目前封装库有许多,下面几个只是star比较多,其中jiecao库比较类似。
ijkplayer官方库
https://github.com/Bilibili/ijkplayer
Vitamio官方库
https://github.com/yixia/VitamioBundle
以jiecao为例的封装库
https://github.com/JasonChow1989/JieCaoVideoPlayer-develop 2年前
https://github.com/open-android/JieCaoVideoPlayer 1年前
https://github.com/lipangit/JiaoZiVideoPlayer 4个月前
https://github.com/CarGuo/GSYVideoPlayer
其他库
https://github.com/danylovolokh/VideoPlayerManager
-
1.2.2 具有的优势
- A.代码布局更加简洁,而且无多余代码
- B.几乎没有多少淡黄色警告,关于注释,通过使用阿里编码插件检测后更加规范,我对代码有洁癖
- C.视频播放器[负责播放],视频控制器[负责视频播放各种点击或者属性设置操作],控制器抽象类[定义属性抽象类,供子类实现],其他可以看代码。结构分层上比较清晰
- D.几乎所有的方法或者重要的成员或者局部变量都有相关的注释,注释的内容非常详细
- E.关于视频属性设置或者按钮点击事件,都可以通过设置相关方法灵活实现。
- 首先这些库封装的思路和代码都不错,我也是借鉴他们的思路,在他们的思路上改进而封装的。
- 相比来说代码结构更加清晰,举几个例子
- 针对视频播放页面布局,由于视频播放状态众多,我封装这库不同状态布局有十几种,许多库的视图布局没注释,显示比较臃肿,如果修改或者定位,不熟悉或者好久不操作,都要花时间找。展示我的布局代码
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<!--https://github.com/yangchong211-->
<!--如果你觉得好,请给个star,让更多人使用,避免重复造轮子-->
<!--底图,主要是显示视频缩略图-->
<ImageView
android:id="@+id/image"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:scaleType="fitXY"
android:visibility="visible"/>
<!--加载动画view-->
<include layout="@layout/custom_video_player_loading"/>
<!--改变播放位置-->
<include layout="@layout/custom_video_player_change_position"/>
<!--改变亮度-->
<include layout="@layout/custom_video_player_change_brightness"/>
<!--改变声音-->
<include layout="@layout/custom_video_player_change_volume"/>
<!--播放完成,你也可以自定义-->
<include layout="@layout/custom_video_player_completed"/>
<!--播放错误-->
<include layout="@layout/custom_video_player_error"/>
<!--顶部控制区-->
<include layout="@layout/custom_video_player_top"/>
<!--底部控制区-->
<include layout="@layout/custom_video_player_bottom"/>
<!--右下角初始显示的总时长-->
<TextView
android:id="@+id/length"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignParentBottom="true"
android:layout_alignParentEnd="true"
android:layout_marginBottom="12dp"
android:layout_marginEnd="8dp"
android:padding="4dp"
android:visibility="visible"
android:text="00:00"
android:textColor="@android:color/white"
android:textSize="12sp"/>
<!--中间开始播放按钮-->
<ImageView
android:id="@+id/center_start"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:src="@drawable/ic_player_center_start"
android:visibility="visible"/>
<!--试看按钮-->
<ImageView
android:id="@+id/iv_try_see"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:src="@drawable/selector_try_see"
android:visibility="gone"/>
<!--试看布局,非会员显示该布局-->
<include layout="@layout/custom_video_player_try_see"/>
</RelativeLayout>
2.关于使用方法说明
2.1 关于gradle引用说明
- 2.1.1直接引用这段代码就可以
compile 'cn.yc:YCVideoPlayerLib:2.2'
2.2 添加布局
- 注意,在实际开发中,由于Android手机碎片化比较严重,分辨率太多了,建议灵活设置布局的宽高比为4:3或者16:9或者你认为合适的,可以用代码设置。
- 如果宽高比变形,则会有黑边
<org.yczbj.ycvideoplayerlib.VideoPlayer
android:id="@+id/video_player"
android:layout_width="match_parent"
android:layout_height="240dp"/>
2.3 最简单的视频播放器参数设定
- 2.3.1 这个是最简单视频播放器的设置参数代码
//设置播放类型
// IjkPlayer or MediaPlayer
videoPlayer1.setPlayerType(VideoPlayer.TYPE_NATIVE);
//网络视频地址
String videoUrl = DataUtil.getVideoListData().get(0).getVideoUrl();
//设置视频地址和请求头部
videoPlayer1.setUp(videoUrl, null);
//是否从上一次的位置继续播放
videoPlayer1.continueFromLastPosition(true);
//设置播放速度
videoPlayer1.setSpeed(1.0f);
//创建视频控制器
VideoPlayerController controller = new VideoPlayerController(this);
controller.setTitle("办快来围观拉,自定义视频播放器可以播放视频拉");
//设置视频时长
controller.setLength(98000);
//设置5秒不操作后则隐藏头部和底部布局视图
controller.setHideTime(5000);
//controller.setImage(R.drawable.image_default);
ImageUtil.loadImgByPicasso(this, R.drawable.image_default, R.drawable.image_default, controller.imageView());
//设置视频控制器
videoPlayer1.setController(controller);
- 2.3.2 关于模仿爱奇艺登录会员权限功能代码
//设置视频加载缓冲时加载窗的类型,多种类型
controller.setLoadingType(2);
ArrayList<String> content = new ArrayList<>();
content.add("试看结束,yc观看全部内容请开通会员1111。");
content.add("试看结束,yc观看全部内容请开通会员2222。");
content.add("试看结束,yc观看全部内容请开通会员3333。");
content.add("试看结束,yc观看全部内容请开通会员4444。");
controller.setMemberContent(content);
controller.setHideTime(5000);
//设置设置会员权限类型,第一个参数是否登录,第二个参数是否有权限看,第三个参数试看完后展示的文字内容,第四个参数是否保存进度位置
controller.setMemberType(false,false,3,true);
controller.imageView().setBackgroundResource(R.color.blackText);
//ImageUtil.loadImgByPicasso(this, R.color.blackText, R.drawable.image_default, controller.imageView());
//设置试看结束后,登录或者充值会员按钮的点击事件
controller.setOnMemberClickListener(new OnMemberClickListener() {
@Override
public void onClick(int type) {
switch (type){
case ConstantKeys.Gender.LOGIN:
//调到用户登录也米娜
startActivity(MeLoginActivity.class);
break;
case ConstantKeys.Gender.MEMBER:
//调到用户充值会员页面
startActivity(MeMemberActivity.class);
break;
default:
break;
}
}
});
- 2.3.3其他设置,让体验更好
- 如果是在Activity中的话,建议设置下面这段代码
@Override
protected void onStop() {
super.onStop();
VideoPlayerManager.instance().releaseVideoPlayer();
}
@Override
public void onBackPressed() {
if (VideoPlayerManager.instance().onBackPressed()) return;
super.onBackPressed();
}
- 如果是在Fragment中的话,建议设置下面这段代码
//在宿主Activity中设置代码如下
@Override
protected void onStop() {
super.onStop();
VideoPlayerManager.instance().releaseVideoPlayer();
}
@Override
public void onBackPressed() {
if (VideoPlayerManager.instance().onBackPressed()) return;
super.onBackPressed();
}
//--------------------------------------------------
//在此Fragment中设置代码如下
@Override
public void onStop() {
super.onStop();
VideoPlayerManager.instance().releaseVideoPlayer();
}
2.4 注意的问题
- 2.4.1如果是全屏播放,则需要在清单文件中设置当前activity的属性值
- android:configChanges 保证了在全屏的时候横竖屏切换不会执行Activity的相关生命周期,打断视频的播放
- android:screenOrientation 固定了屏幕的初始方向
- 这两个变量控制全屏后和退出全屏的屏幕方向
<activity android:name=".ui.test2.TestMyActivity"
android:configChanges="orientation|keyboardHidden|screenSize"
android:screenOrientation="portrait"/>
2.5 关于开源库中的类说明
3.关于播放类型说明
3.1 普通视频播放
- 3.1.1 这一步操作可以直接看第二部分内容——关于使用方法说明
3.2 list页面视频播放
- 3.2.1如何在list页面设置视频
- 第一步:在activity或者fragment中
recyclerView.setLayoutManager(new LinearLayoutManager(this));
recyclerView.setHasFixedSize(true);
VideoAdapter adapter = new VideoAdapter(this, DataUtil.getVideoListData());
recyclerView.setAdapter(adapter);
//注意:下面这个方法不能漏掉
recyclerView.setRecyclerListener(new RecyclerView.RecyclerListener() {
@Override
public void onViewRecycled(RecyclerView.ViewHolder holder) {
VideoPlayer videoPlayer = ((VideoAdapter.VideoViewHolder) holder).mVideoPlayer;
if (videoPlayer == VideoPlayerManager.instance().getCurrentVideoPlayer()) {
VideoPlayerManager.instance().releaseVideoPlayer();
}
}
});
- 第二步:在RecyclerView的适配器Adapter中
public class VideoAdapter extends RecyclerView.Adapter<VideoAdapter.VideoViewHolder> {
private Context mContext;
private List<Video> mVideoList;
VideoAdapter(Context context, List<Video> videoList) {
mContext = context;
mVideoList = videoList;
}
@Override
public VideoViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View itemView = LayoutInflater.from(mContext).inflate(R.layout.item_test_my_video, parent, false);
VideoViewHolder holder = new VideoViewHolder(itemView);
//创建视频播放控制器,主要只要创建一次就可以呢
VideoPlayerController controller = new VideoPlayerController(mContext);
holder.setController(controller);
return holder;
}
@Override
public void onBindViewHolder(VideoViewHolder holder, int position) {
Video video = mVideoList.get(position);
holder.bindData(video);
}
@Override
public int getItemCount() {
return mVideoList==null ? 0 : mVideoList.size();
}
class VideoViewHolder extends RecyclerView.ViewHolder {
VideoPlayerController mController;
VideoPlayer mVideoPlayer;
VideoViewHolder(View itemView) {
super(itemView);
mVideoPlayer = (VideoPlayer) itemView.findViewById(R.id.nice_video_player);
// 将列表中的每个视频设置为默认16:9的比例
ViewGroup.LayoutParams params = mVideoPlayer.getLayoutParams();
// 宽度为屏幕宽度
params.width = itemView.getResources().getDisplayMetrics().widthPixels;
// 高度为宽度的9/16
params.height = (int) (params.width * 9f / 16f);
mVideoPlayer.setLayoutParams(params);
}
/**
* 设置视频控制器参数
* @param controller 控制器对象
*/
void setController(VideoPlayerController controller) {
mController = controller;
mVideoPlayer.setController(mController);
}
void bindData(Video video) {
mController.setTitle(video.getTitle());
mController.setLength(video.getLength());
Glide.with(itemView.getContext())
.load(video.getImageUrl())
.placeholder(R.drawable.image_default)
.crossFade()
.into(mController.imageView());
mVideoPlayer.setUp(video.getVideoUrl(), null);
}
}
}
3.3 小窗口视频播放
- 3.3.1建议在设置小窗口先先判断视频播放器是否开始播放
if (videoPlayer.isIdle()) {
Toast.makeText(this, "要点击播放后才能进入小窗口", Toast.LENGTH_SHORT).show();
} else {
videoPlayer.enterTinyWindow();
}
3.4 类似爱奇艺,优酷会员试看视频播放
- 3.4.1 可以参考——2.3.2 关于模仿爱奇艺登录会员权限功能代码
3.5 关于封装库中日志打印
- 3.5.1关于封装库中日志打印设置
- 如果上线产品后不想打印日志,可以在初始化时设置,注意需要在初始化播放器之前设置
//如果不想打印库中的日志,可以设置
VideoLogUtil.isLog = false;
- 3.5.3关于日志工具类代码
public class VideoLogUtil {
private static final String TAG = "YCVideoPlayer";
public static boolean isLog = true;
static void d(String message) {
if(isLog){
Log.d(TAG, message);
}
}
static void i(String message) {
if(isLog){
Log.i(TAG, message);
}
}
static void e(String message, Throwable throwable) {
if(isLog){
Log.e(TAG, message, throwable);
}
}
}
4.关于相关方法说明
4.1 关于VideoPlayer类中方法说明
- 4.1.1 关于一定需要这四步
//设置播放类型
// IjkPlayer or MediaPlayer
videoPlayer1.setPlayerType(VideoPlayer.TYPE_NATIVE);
//设置视频地址和请求头部
videoPlayer1.setUp(videoUrl, null);
//创建视频控制器
VideoPlayerController controller = new VideoPlayerController(this);
//设置视频控制器
videoPlayer1.setController(controller);
- 4.1.2 关于VideoPlayer中设置属性方法
//设置播放类型
// MediaPlayer
videoPlayer.setPlayerType(VideoPlayer.TYPE_NATIVE);
// IjkPlayer
videoPlayer.setPlayerType(VideoPlayer.TYPE_IJK);
//网络视频地址
String videoUrl = DataUtil.getVideoListData().get(1).getVideoUrl();
//设置视频地址和请求头部
videoPlayer.setUp(videoUrl, null);
//是否从上一次的位置继续播放
videoPlayer.continueFromLastPosition(false);
//设置播放速度
videoPlayer.setSpeed(1.0f);
//设置播放位置
//videoPlayer.seekTo(3000);
//设置音量
videoPlayer.setVolume(50);
//设置全屏播放
videoPlayer.enterFullScreen();
//设置小屏幕播放
videoPlayer.enterTinyWindow();
//退出全屏
videoPlayer.exitFullScreen();
//退出小窗口播放
videoPlayer.exitTinyWindow();
//释放,内部的播放器被释放掉,同时如果在全屏、小窗口模式下都会退出
videoPlayer.release();
//释放播放器,注意一定要判断对象是否为空,增强严谨性
videoPlayer.releasePlayer();
- 4.1.3 关于VideoPlayer中获取属性方法
//是否从上一次的位置继续播放,不必须
videoPlayer.continueFromLastPosition(false);
//获取最大音量
int maxVolume = videoPlayer.getMaxVolume();
//获取音量值
int volume = videoPlayer.getVolume();
//获取持续时长
long duration = videoPlayer.getDuration();
//获取播放位置
long currentPosition = videoPlayer.getCurrentPosition();
//获取缓冲区百分比
int bufferPercentage = videoPlayer.getBufferPercentage();
//获取播放速度
float speed = videoPlayer.getSpeed(1);
- 4.1.4 关于VideoPlayer中设置播放状态方法
//开始播放
videoPlayer.start();
//开始播放,从某位置播放
videoPlayer.start(3000);
//重新播放
videoPlayer.restart();
//暂停播放
videoPlayer.pause();
- 4.1.5 关于VideoPlayer中获取播放状态方法
//判断是否开始播放
boolean idle = videoPlayer.isIdle();
//判断视频是否播放准备中
boolean preparing = videoPlayer.isPreparing();
//判断视频是否准备就绪
boolean prepared = videoPlayer.isPrepared();
//判断视频是否正在缓冲
boolean bufferingPlaying = videoPlayer.isBufferingPlaying();
//判断是否是否缓冲暂停
boolean bufferingPaused = videoPlayer.isBufferingPaused();
//判断视频是否暂停播放
boolean paused = videoPlayer.isPaused();
//判断视频是否正在播放
boolean playing = videoPlayer.isPlaying();
//判断视频是否播放错误
boolean error = videoPlayer.isError();
//判断视频是否播放完成
boolean completed = videoPlayer.isCompleted();
//判断视频是否播放全屏
boolean fullScreen = videoPlayer.isFullScreen();
//判断视频是否播放小窗口
boolean tinyWindow = videoPlayer.isTinyWindow();
//判断视频是否正常播放
boolean normal = videoPlayer.isNormal();
4.2 关于VideoPlayerController类[控制器]中方法说明
- 4.2.1 关于控制器方法
//创建视频控制器
VideoPlayerController controller = new VideoPlayerController(this);
//设置视频标题
controller.setTitle("高仿优酷视频播放页面");
//设置视频时长
//controller.setLength(98000);
//设置视频加载缓冲时加载窗的类型,多种类型
controller.setLoadingType(2);
ArrayList<String> content = new ArrayList<>();
content.add("试看结束,观看全部内容请开通会员1111。");
content.add("试看结束,观看全部内容请开通会员2222。");
content.add("试看结束,观看全部内容请开通会员3333。");
content.add("试看结束,观看全部内容请开通会员4444。");
//设置会员权限话术内容
controller.setMemberContent(content);
//设置不操作后,5秒自动隐藏头部和底部布局
controller.setHideTime(5000);
//设置设置会员权限类型,第一个参数是否登录,第二个参数是否有权限看,第三个参数试看完后展示的文字内容,第四个参数是否保存进度位置
controller.setMemberType(false,false,3,true);
//设置背景图片
controller.imageView().setBackgroundResource(R.color.blackText);
//ImageUtil.loadImgByPicasso(this, R.color.blackText, R.drawable.image_default, controller.imageView());
//设置试看结束后,登录或者充值会员按钮的点击事件
controller.setOnMemberClickListener(new OnMemberClickListener() {
@Override
public void onClick(int type) {
switch (type){
case ConstantKeys.Gender.LOGIN:
//调到用户登录也米娜
startActivity(MeLoginActivity.class);
break;
case ConstantKeys.Gender.MEMBER:
//调到用户充值会员页面
startActivity(MeMemberActivity.class);
break;
default:
break;
}
}
});
//设置视频清晰度
//videoPlayer.setClarity(list,720);
//设置视频控制器
videoPlayer.setController(controller);
4.3 关于对象的销毁
- 4.3.1在VideoPlayer中如何释放资源的呢?源代码如下所示
@Override
public void release() {
// 保存播放位置
if (isPlaying() || isBufferingPlaying() || isBufferingPaused() || isPaused()) {
VideoPlayerUtils.savePlayPosition(mContext, mUrl, getCurrentPosition());
} else if (isCompleted()) {
//如果播放完成,则保存播放位置为0,也就是初始位置
VideoPlayerUtils.savePlayPosition(mContext, mUrl, 0);
}
// 退出全屏或小窗口
if (isFullScreen()) {
exitFullScreen();
}
if (isTinyWindow()) {
exitTinyWindow();
}
mCurrentMode = MODE_NORMAL;
// 释放播放器
releasePlayer();
// 恢复控制器
if (mController != null) {
mController.reset();
}
// gc回收
Runtime.getRuntime().gc();
}
//释放播放器,注意一定要判断对象是否为空,增强严谨性
@Override
public void releasePlayer() {
if (mAudioManager != null) {
//放弃音频焦点。使以前的焦点所有者(如果有的话)接收焦点。
mAudioManager.abandonAudioFocus(null);
//置空
mAudioManager = null;
}
if (mMediaPlayer != null) {
//释放视频焦点
mMediaPlayer.release();
mMediaPlayer = null;
}
//从视图中移除TextureView
mContainer.removeView(mTextureView);
if (mSurface != null) {
mSurface.release();
mSurface = null;
}
//如果SurfaceTexture不为null,则释放
if (mSurfaceTexture != null) {
mSurfaceTexture.release();
mSurfaceTexture = null;
}
//设置状态
mCurrentState = STATE_IDLE;
}
5.关于封装的思路
5.1 参考的案例思路
- 5.1.1目前参考的案例有
- 可以直接看下面的参考案例,有记录
- 5.1.2针对jiaozi代码简单分析
- JZVideoPlayer为继承自FrameLayout实现的一个组合自定义View来实现了视频播放器的View相关的内容。
- JZVideoPlayerStandard则是继承自JZVideoPlayer实现了一些自身的功能。
- JZMediaManager是用来对于MediaPlayer的管理,对于MediaPlayer的一些监听器方法的回调和TextrueView的相关回调处理。
- JZVideoPlayerManager管理JZVideoPlayer
- 和自定义相关的工作,最主要是先继承JCVideoPlayerStandard
- JZMediaSystem主要是实现系统的播放引擎
- 不得不说,大神封装代码的思路以及代码逻辑的确很强
- 关于封装库其他感受
- 第一:不过,感觉大神更新频率大高,而且没有找到每次更新的日志说明,不知道大神又解决了那些bug
- 第二:黄色警告多,而且注释少,因为视频封装库不像一般库,有时候需求不同,可拓展性要求高。除了自己继承JCVideoPlayerStandard创建视频播放器,其他如果想改代码,还是有点复杂的。
- 第三:关于使用虽然很简单,但是在JZVideoPlayerStandard这个方法中,布局的对象都是用public修饰,如果你要想自己甚至某个控件背景或者图标等等,则要这样应用。如果你不去看看源代码中布局名称,你根本就不知道这个对象对应的是什么东西。对于不同修饰符,要合适的,如果不合适,那么就会有淡黄色警告。我看了buttonKnife,retrofit,阿里vlayout等等,可以说黄色警告很少……
Picasso.with(this)
.load("http://jzvd-pic.nathen.cn/jzvd-pic/1bb2ebbe-140d-4e2e-abd2-9e7e564f71ac.png")
.into(jzVideo.thumbImageView);
5.2 封装的基本思路
- 5.2.1关于简单的思路分析
- a1.可以把视频播放和设置视频属性控制器分离,对于VideoPlayer中,各种UI状态和操作反馈都封装到VideoPlayerController控制器里面。如果需要根据不同的项目需求来修改播放器的功能,就只重写VideoPlayerController就可以了。
- a2.对于VideoPlayer这个类,可以先创建一个帧布局容器,然后在初始化的时候将视频播放器控制器放到里面,然后通过设置控制器来进行视频播放
- a3.当调用了开始播放的方法后,就初始化播放器,包括原生的,还有IjkMediaPlayer
- a4.而基于IjkMediaPlayer的视频播放,需要添加各种监听事件,通过阅读IMediaPlayer源码可以知道:可以在这些监听事件中添加各种对视频的操作逻辑,具体可以看代码。
void setOnPreparedListener(IMediaPlayer.OnPreparedListener var1);
void setOnCompletionListener(IMediaPlayer.OnCompletionListener var1);
void setOnBufferingUpdateListener(IMediaPlayer.OnBufferingUpdateListener var1);
void setOnSeekCompleteListener(IMediaPlayer.OnSeekCompleteListener var1);
void setOnVideoSizeChangedListener(IMediaPlayer.OnVideoSizeChangedListener var1);
void setOnErrorListener(IMediaPlayer.OnErrorListener var1);
void setOnInfoListener(IMediaPlayer.OnInfoListener var1);
void setOnTimedTextListener(IMediaPlayer.OnTimedTextListener var1);
- a5.定义好了监听事件后,就创建了播放,重置播放,暂停等各种方法
5.3 关于窗口切换分析
- 5.3.1 关于窗口切换调用的代码
//设置全屏播放
videoPlayer.enterFullScreen();
//设置小屏幕播放
videoPlayer.enterTinyWindow();
//退出全屏
videoPlayer.exitFullScreen();
//退出小窗口播放
videoPlayer.exitTinyWindow();
//释放,内部的播放器被释放掉,同时如果在全屏、小窗口模式下都会退出
videoPlayer.release();
//释放播放器,注意一定要判断对象是否为空,增强严谨性
videoPlayer.releasePlayer();
5.4 关于VideoPlayerManager视频播放器管理器分析
- 5.4.1可以直接看源代码,我对每个方法都有详细的注释
public class VideoPlayerManager {
private VideoPlayer mVideoPlayer;
private static VideoPlayerManager sInstance;
private VideoPlayerManager() {}
//一定要使用单例模式,保证同一时刻只有一个视频在播放,其他的都是初始状态
public static synchronized VideoPlayerManager instance() {
if (sInstance == null) {
sInstance = new VideoPlayerManager();
}
return sInstance;
}
public VideoPlayer getCurrentVideoPlayer() {
return mVideoPlayer;
}
void setCurrentVideoPlayer(VideoPlayer videoPlayer) {
if (mVideoPlayer != videoPlayer) {
releaseVideoPlayer();
mVideoPlayer = videoPlayer;
}
}
//当视频正在播放或者正在缓冲时,调用该方法暂停视频
public void suspendVideoPlayer() {
if (mVideoPlayer != null && (mVideoPlayer.isPlaying() || mVideoPlayer.isBufferingPlaying())) {
mVideoPlayer.pause();
}
}
//当视频暂停时或者缓冲暂停时,调用该方法重新开启视频播放
public void resumeVideoPlayer() {
if (mVideoPlayer != null && (mVideoPlayer.isPaused() || mVideoPlayer.isBufferingPaused())) {
mVideoPlayer.restart();
}
}
//释放,内部的播放器被释放掉,同时如果在全屏、小窗口模式下都会退出
public void releaseVideoPlayer() {
if (mVideoPlayer != null) {
mVideoPlayer.release();
mVideoPlayer = null;
}
}
//处理返回键逻辑.如果是全屏,则退出全屏 如果是小窗口,则退出小窗口
public boolean onBackPressed() {
if (mVideoPlayer != null) {
if (mVideoPlayer.isFullScreen()) {
return mVideoPlayer.exitFullScreen();
} else if (mVideoPlayer.isTinyWindow()) {
return mVideoPlayer.exitTinyWindow();
}
}
return false;
}
}
5.5 关于VideoPlayerController视频控制器分析
- 5.5.1VideoPlayerController的作用
- 播放控制界面上,播放、暂停、播放进度、缓冲动画、全屏/小屏等触发都是直接调用播放器对应的操作的。
- 5.5.2VideoPlayerController的方法如下所示
//创建视频控制器
VideoPlayerController controller = new VideoPlayerController(this);
//设置视频标题
controller.setTitle("高仿优酷视频播放页面");
//设置视频时长
//controller.setLength(98000);
//设置视频加载缓冲时加载窗的类型,多种类型
controller.setLoadingType(2);
ArrayList<String> content = new ArrayList<>();
content.add("试看结束,观看全部内容请开通会员1111。");
content.add("试看结束,观看全部内容请开通会员2222。");
content.add("试看结束,观看全部内容请开通会员3333。");
content.add("试看结束,观看全部内容请开通会员4444。");
//设置会员权限话术内容
controller.setMemberContent(content);
//设置不操作后,5秒自动隐藏头部和底部布局
controller.setHideTime(5000);
//设置设置会员权限类型,第一个参数是否登录,第二个参数是否有权限看,第三个参数试看完后展示的文字内容,第四个参数是否保存进度位置
controller.setMemberType(false,false,3,true);
//设置背景图片
controller.imageView().setBackgroundResource(R.color.blackText);
//ImageUtil.loadImgByPicasso(this, R.color.blackText, R.drawable.image_default, controller.imageView());
//设置试看结束后,登录或者充值会员按钮的点击事件
controller.setOnMemberClickListener(new OnMemberClickListener() {
@Override
public void onClick(int type) {
switch (type){
case ConstantKeys.Gender.LOGIN:
//调到用户登录也米娜
startActivity(MeLoginActivity.class);
break;
case ConstantKeys.Gender.MEMBER:
//调到用户充值会员页面
startActivity(MeMemberActivity.class);
break;
default:
break;
}
}
});
//设置视频清晰度
//videoPlayer.setClarity(list,720);
//设置视频控制器
videoPlayer.setController(controller);
5.6 关于InterVideoPlayer接口分析
- 5.6.1关于此接口方法有
- 跟jiaozi代码类似
/**
* 设置视频Url,以及headers
*
* @param url 视频地址,可以是本地,也可以是网络视频
* @param headers 请求header.
*/
void setUp(String url, Map<String, String> headers);
/**
* 开始播放
*/
void start();
/**
* 从指定的位置开始播放
*
* @param position 播放位置
*/
void start(long position);
/**
* 重新播放,播放器被暂停、播放错误、播放完成后,需要调用此方法重新播放
*/
void restart();
/**
* 暂停播放
*/
void pause();
/**
* seek到制定的位置继续播放
*
* @param pos 播放位置
*/
void seekTo(long pos);
/**
* 设置音量
*
* @param volume 音量值
*/
void setVolume(int volume);
/**
* 设置播放速度,目前只有IjkPlayer有效果,原生MediaPlayer暂不支持
*
* @param speed 播放速度
*/
void setSpeed(float speed);
/**
* 开始播放时,是否从上一次的位置继续播放
*
* @param continueFromLastPosition true 接着上次的位置继续播放,false从头开始播放
*/
void continueFromLastPosition(boolean continueFromLastPosition);
6.关于如何自定义你想要的视频播放模式
6.1 自定义视频播放器
- 6.1.1如何自定义自己的播放器
- 第一步:首先继承VideoPlayer这个类
- 第二步:然后重写部分你需要更改功能的方法,只需要选择你需要重写的方法即可。
- 6.1.2代码展示如下所示
public class YCVideoPlayer extends VideoPlayer {
public YCVideoPlayer(Context context) {
super(context);
}
@Override
public void setUp(String url, Map<String, String> headers) {
super.setUp(url, headers);
}
@Override
public void setController(AbsVideoPlayerController controller) {
super.setController(controller);
}
@Override
public void setPlayerType(int playerType) {
super.setPlayerType(playerType);
}
@Override
public void continueFromLastPosition(boolean continueFromLastPosition) {
super.continueFromLastPosition(continueFromLastPosition);
}
@Override
public void setSpeed(float speed) {
super.setSpeed(speed);
}
@Override
public void start() {
super.start();
}
@Override
public void start(long position) {
super.start(position);
}
@Override
public void restart() {
super.restart();
}
@Override
public void pause() {
super.pause();
}
@Override
public void seekTo(long pos) {
super.seekTo(pos);
}
@Override
public void setVolume(int volume) {
super.setVolume(volume);
}
@Override
public boolean isIdle() {
return super.isIdle();
}
@Override
public boolean isPreparing() {
return super.isPreparing();
}
@Override
public boolean isPrepared() {
return super.isPrepared();
}
@Override
public boolean isBufferingPlaying() {
return super.isBufferingPlaying();
}
@Override
public boolean isBufferingPaused() {
return super.isBufferingPaused();
}
@Override
public boolean isPlaying() {
return super.isPlaying();
}
@Override
public boolean isPaused() {
return super.isPaused();
}
@Override
public boolean isError() {
return super.isError();
}
}
7.关于效果图的展示
7.1 效果图如下所示
8.关于遇到的问题说明
8.1 视频难点
- 8.1.1 当视频切换全屏或者从全屏切换到正常小屏幕时,如何管理activity的生命周期
- 8.1.2 在列表list页面,滑动显示小窗口,那么什么时候显示小窗口呢?关于RecyclerView的滑动位移超出屏幕有没有更好的解决办法?
- 8.1.2 当屏幕从全屏退出时,播放位置要滑到记录的位置,代码逻辑复杂,如何避免耦合度太高
8.2 遇到的bug
- 8.2.1 当视频切花时,如何避免视频不卡顿
- 8.2.2 在fragment中,当左右滑动出另一个fragment中,视频还在播放,怎么样处理这部分逻辑
- 8.2.3 在显示缓冲比时,网络不好或者暂停缓冲时有问题,所以暂停还没有添加该功能
- 8.2.4 播放进度条seekbar跳动问题,有人反映不是那么顺畅
- 8.2.5 部分华为手机播放视频有问题,在找原因
- 8.2.6 在拖动时显示当前帧的画面图片,类似优酷那个功能,最终还是没有实现
8.3 后期需要实现的功能
- 8.3.1 如果有多集视频,则添加上一集和下一集的功能
- 8.3.2 拖动滑动条,显示帧画面
- 8.3.3 实现弹幕功能
- 8.4.4 有些手机播放有问题,测试找问题
- 8.5.5 切换视频清晰度有问题,是重新开始播放,因为切换清晰度时,调用的视频链接是不同的。比如高清视频和标准视频链接是不同的,所以难以实现切换后记录位置播放。但是看了下优酷,爱奇艺视频,切换后是接着之前观看的位置播放,这个需要思考下怎么实现。欢迎同行给出好的建议。
- 8.5.6 待定
9.关于版本更新说明
- 9.1 V1.0.0 更新于2017年10月4日
- 9.2 V1.0.1 更新于2017年11月18日
- 9.3 v1.1.0 更新于2018年1月15日
10.关于参考文档说明
10.1 参考的项目
- 10.1.1参考的开源项目有
https://github.com/CarGuo/GSYVideoPlayer
https://github.com/danylovolokh/VideoPlayerManager
https://github.com/HotBitmapGG/bilibili-android-client
https://github.com/jjdxmashl/jjdxm_ijkplayer
https://github.com/JasonChow1989/JieCaoVideoPlayer-develop 2年前
https://github.com/open-android/JieCaoVideoPlayer 1年前
https://github.com/lipangit/JiaoZiVideoPlayer 4个月前
https://github.com/xiaoyanger0825/NiceVieoPlayer
https://github.com/curtis2/SuperVideoPlayer
https://github.com/tcking/GiraffePlayer
10.2 参考的博客
- 10.2.1参考的博客有'
https://segmentfault.com/a/1190000011959615
http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2014/1213/2153.html
http://blog.csdn.net/junwang19891012/article/details/8444743
https://www.jianshu.com/p/420f7b14d6f6
http://blog.csdn.net/candicelijx/article/details/39495271
11.关其他说明
11.1 目前市场流行的视频框架
- 1.Android原生VideoView
- 2.Google 开源视频播放框架 ExoPlayer
- 3.Vitamio 视频播放框架
- 4.Bilibili 开源视频播放框架ijkplayer
11.2 如何选择
- 11.2.1.Android原生VideoView
-
1.1 VideoView 的使用非常简单,播放视频的步骤:
- 在界面布局文件中定义 VideoView 组件,或在程序中创建 VideoView 组件
-
调用 VideoView 的如下两个方法来加载指定的视频:
- setVidePath(String path):加载 path 文件代表的视频
- setVideoURI(Uri uri):加载 uri 所对应的视频
- 调用 VideoView 的 start()、stop()、psuse() 方法来控制视频的播放
- 11.2.2.Google 开源视频播放框架 ExoPlayer
- 2.1 框架地址:https://github.com/google/Exo...
-
2.2 用法
-
ExoPlayer 开源项目包含了 library 和 示例:
- ExoPlayer library – 这部分是核心的库
- Demo app – 这部分是演示怎么使用 ExoPlayer 的 Demo
- ExoPlayer 库的核心类是 ExoPlayer 类。该类维护了播放器的全局状态 。比如如何获取媒体数据,如何缓冲以及是怎样的编码格式。
- ExoPlayer 基于 MediaCodec 和 AudioTrack 提供了默认的音视频的 TrackRenderer 实现。所有的 renderers 都需要 SampleSource 对象,ExoPlayer 从 SampleSource 获得 media samples 用于播放。下图展示了 ExoPlayer 是如何配置组合这些组件用于播放音视频的。
- standard-model
- ExoPlayer 库提供了一些不同类型的 SampleSource 实例:
-
ExtractorSampleSource – 用于 MP3,M4A,WebM,MPEG-TS 和 AAC;
- ChunkSampleSource – 用于 DASH 和平滑流的播放;
- HlsSampleSource – 用于 HLS 播放;
-
在 ExoPlayer 的 Dome 中使用 DemoPlayer 对 ExoPlayer 进行了封装,并提供了使用上述几种 SampleSource 构建 TrackRenderer 的 Builder。
- SmoothStreamingRendererBuilder
- DashRendererBuilder
- ExtractorRendererBuilder
- 在使用的时候我们根据不同的需求创建对应的 RendererBuilder,然后将 RendererBuilder 传递给 DemoPlayer 然后调用 DemoPlayer 的 setPlayWhenReady 方法。
-
- 2.3 优缺点
-
ExoPlayer 相较于 MediaPlayer 有很多很多的优点:
- 支持动态的自适应流 HTTP (DASH) 和 平滑流,任何目前 MediaPlayer 支持的视频格式(同时它还支持 HTTP 直播(HLS),MP4,MP3,WebM,M4A,MPEG-TS 和 AAC)。
- 支持高级的 HLS 特性,例如正确处理 EXT-X-DISCONTINUITY 标签;
- 支持自定义和扩治你的使用场景。ExoPlayer 专门为此设计;
- 便于随着 App 的升级而升级。因为 ExoPlayer 是一个包含在你的应用中的库,对于你使用哪个版本有完全的控制权,并且你可以简单的跟随应用的升级而升级;
- 更少的适配性问题。
-
ExoPlayer 的缺点:
- ExoPlayer 的音频和视频组件依赖 Android 的 MediaCodec 接口,该接口发布于 Android4.1(API 等级 16)。因此它不能工作于之前的Android 版本。
- 11.2.3.Vitamio 视频播放框架
- 3.1 用法
- 官网:https://www.vitamio.org
-
Vitamio 的使用步骤:
- 1.下载 Vitamio 库,并作为工程依赖。
- 2.在 Activity 的 onCreate 方法中添加如下代码,初始化 Vitamio 的解码器
-
3.2 优点
- 强大,支持超多格式视频和网络视频播放。
- 使用简单。调用非常简单,方便使用。
- 其官方还给出了其他很多优点,但是个人觉得不足以成为优点。
- 11.2.4.Bilibili 开源视频播放框架ijkplayer
-
4.1 特点
- HTTPS支持
- 支持弹幕
- 支持基本的拖动,声音、亮度调节
- 支持边播边缓存
- 支持视频本身自带rotation的旋转(90,270之类),重力旋转与手动旋转的同步支持
- 支持列表播放,直接添加控件为封面,列表全屏动画,视频加载速度,列表小窗口支持拖动
- 5.0的过场效果,调整比例,多分辨率切换
- 支持切换播放器,进度条小窗口预览
- 其他一些小动画效果,rtsp、concat、mpeg
-
4.2 优缺点
- ijkplayer 最大的优点就是可以根据需要编译需要的解码器。在编译的时候通过 ln -s module-default.sh module.sh 选择要编译的解码器。ijkplayer 在 config 目录下提供了三种 module.sh 。也可自己修改 module.sh 。
- ijkplayer 的缺点是库太大。加入项目后会大大增加你的 APP 的大小。
11.3 关于我的个人博客和站点
- github: https://github.com/yangchong211
- 知乎: https://www.zhihu.com/people/yang-chong-69-24/pins/posts
- 简书: http://www.jianshu.com/u/b7b2c6ed9284
- csdn: http://my.csdn.net/m0_37700275
- 喜马拉雅听书: http://www.ximalaya.com/zhubo/71989305/
- 泡在网上的日子:http://www.jcodecraeer.com/member/content_list.php?channelid=1
- 邮箱:yangchong211@163.com
- 阿里云博客:https://yq.aliyun.com/users/article?spm=5176.100239.headeruserinfo.3.dT4bcV
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。