一、背景

    最近维护了一个点播播放器,基于B站开源ijkplayer播放器做二次开发,支持Android和iOS平台。Android平台在Java层对ijkplayer包装了一层,做成了一个sdk的aar包,给App使用;iOS基于object-c将ijkplayer包装了一层,打成framework给App使用。

    同事离职后,接手其所负责事宜,测试同事提了一系列播放缺陷,诸如切到前台概率性黑屏、屏幕旋转画面扭曲或者不继续播放、seek崩溃或时间点向前跳跃几秒、最终播放时长和片源时长不一致、画中画视频不继续播、暂停后全屏画面扭曲等等问题。

    本文选取暂停播放后切到前台概率性黑屏问题进行分析并给出解决办法。

二、问题现象

    如本文主题,App暂停播放后再切到前台概率性黑屏,暂停后全屏也黑屏。

三、分析

1)分析发现,暂停后如果不切到后台而只是切换页面再切回来,经长时间多次观察,不会出现黑屏,画面保留最后一帧,这说明ijkplayer底层在播放暂停后会重绘最后一帧画面

2)但是,切到后台再切回前台时,却出现概率性黑屏问题,说明原因大概率是出在Java这层

四、解决方案

    于是,想到一招规避办法:Java层在暂停时保留最后一帧画面,让它贴到ImgeView控件上,并把ImageView加入到FrameLayout容器里显示,问题得以解决

1)视频初次加载时,调用createVideoView()方法创建IjkVideoView窗口,作为Video窗口,再让其贴在父窗口mVideoGroup上,并设置其随父窗口伸缩适应屏幕,缺省显示该视频窗口。

2)而后,创建ImageView窗口mImageViewWhenPaused,并设置其铺满屏幕mImageViewWhenPaused.setScaleType(ImageView.ScaleType.CENTER_CROP);

Android 播放完黑屏 安卓手机播放视频黑屏_sed

此处,mVideoView视频窗口显示模式mDisplayMode=AR_ASPECT_FIT_PARENT,让其适配屏幕。

以下时ijkplayer所支持的所有屏幕高宽比:

Android 播放完黑屏 安卓手机播放视频黑屏_App_02

3)当用户点击暂停按钮时,保存最后一帧画面,

Android 播放完黑屏 安卓手机播放视频黑屏_android_03

在updateViewForPaused()方法里获取当前帧的位图,并设置给mImageViewWhenPaused,再让该窗口显示可见。
    mImageViewWhenPaused.setImageBitmap(mBitmapWhenPaused);
    mImageViewWhenPaused.setVisibility(VISIBLE);

Android 播放完黑屏 安卓手机播放视频黑屏_sed_04

 4)最后,在点击关闭按钮停止播放时,释放这一帧数据:

Android 播放完黑屏 安卓手机播放视频黑屏_Android 播放完黑屏_05

 通过以上实践,最终可解决android平台暂停后切到前台暂停后seek播放再切到前台画面黑屏的问题。