完整视频播放器封装库,仿优酷

目录介绍

  • 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/yangchong211/YCVideoPlayer

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上项目

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 关于开源库中的类说明

  • image

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 效果图如下所示

image image image image image image image image image image image

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/ExoPlayer

  • 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 关于我的个人博客和站点