Android ijkplayer的使用方法解析

ijkplayer是Bilibili基于ffmpeg开发并开源的轻量级视频播放器,支持播放本地网络视频,也支持流媒体播放。支持Android&iOS。

ijkplayer的编译这里不多阐述,我也是直接获取别人编译完成的so库文件,直接使用的。如果你对ijkplayer的编译感兴趣,可以百度一下,有很多文章。

使用ijkplayer

导包

ijkplayer源码官方下载地址:https://github.com/Bilibili/ijkplayer

上面是官方提供的ijkplayer的源码地址,但是它是没有编译过的。下面我给大家分享一份编译好的ijkplayer源码,由于比较大,分了三个包才上传完成,需要三个包都下载后才能一起解压: ijkplayer_jb51.rar

我们下载完成,进入android/ijkplayer目录:

  • ijkplayer-java:ijkplayer的一些操作封装及定义。这里面是通用的API接口,里面最主要的是IMediaPlayer,它是用来渲染显示多媒体的。
  • ijkplayer-exo:google开源的一个新的播放器ExoPlayer,在Demo中和ijkplayer对比用的。通过安装ijkplayer可以发现setting里面可以选择不同player来渲染多媒体显示,该模块下面就是一个MediaPlayer。
  • ijkplayer-example:测试程序
  • ijkplayer-{arch}:编译出来的各个版本的.so文件。

官方提供的Demo的代码还是挺多的,甚至还用了otto,需要对官方的demo进行精简,去除一些用不到的代码。

首先需要的是ijkplayer-{arch}、ijkplayer-java两个库。exo是Google提供的新的播放器,这里不需要,直接砍掉。其次是ijkplayer-example里的,我们需要的只有tv.danmaku.ijk.media.example.widget.media包下的部分类。

注:

链接库ijkplayer-arm64,ijkplayer-armv5,ijkplayer-armv7a,ijkplayer-x86,ijkplayer-x86_64是不同体系架构的动态链接库,在当前工程结构里面作为一个模块,如果不想做兼容多平台问题,可以删除其他目录结构,单独保留自己需要的平台目录。

新建一个工程:

(1)把ijkplayer-armv7a/src/main/libs下的文件拷贝到新工程app目录的libs下。
(2)把ijkplayer-java/build/outputs/aar/ijkplayer-java-release.aar复制到新工程app目录的libs下。
(3)修改APP下的build.gradle, 主要设置.so及.aar的位置:

apply plugin: 'com.android.application'

android {
  compileSdkVersion 24
  buildToolsVersion "25.0.0"
  defaultConfig {
    applicationId "com.hx.ijkplayer_demo"
    minSdkVersion 14
    targetSdkVersion 24
    versionCode 1
    versionName "1.0"
    testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
  }
  buildTypes {
    release {
      minifyEnabled false
      proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
    }
  }
  sourceSets {
    main {
      jniLibs.srcDirs = ['libs'] /**在libs文件夹下找so文件*/
    }
  }
}

repositories {
  mavenCentral()
  flatDir {
    dirs 'libs' /**在libs文件夹下找aar文件*/
  }
}

dependencies {
  compile fileTree(dir: 'libs', include: ['*.jar'])
  androidTestCompile('com.android.support.test.espresso:espresso-core:2.2.2', {
    exclude group: 'com.android.support', module: 'support-annotations'
  })
  compile 'com.android.support:appcompat-v7:24.2.1'
  testCompile 'junit:junit:4.12'
  compile(name: 'ijkplayer-java-release', ext: 'aar') /**编译ijkplayer-java-release.aar文件*/
}

(4)复制ijkplayer-example下面的tv.danmaku.ijk.media.example.widget.media到新的工程,删掉一些不需要类。

(5)IjkVideoView里面还是有很多如exo等没用的东西,删!具体可以参见我后面的Demo。

(6)Manifest

...
<activity android:name=".MainActivity"
     android:screenOrientation="sensorLandscape"
     android:configChanges="orientation|keyboardHidden">
...
</activity>
...
<uses-permission android:name="android.permission.INTERNET"/>

播放实现

Xml代码

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
  xmlns:android="http://schemas.android.com/apk/res/android"
  xmlns:tools="http://schemas.android.com/tools"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  tools:context=".MainActivity">

  <com.hx.ijkplayer_demo.widget.media.IjkVideoView
    android:id="@+id/video_view"
    android:layout_width="match_parent"
    android:layout_height="match_parent"/>
</RelativeLayout>

Java代码

public class MainActivity extends AppCompatActivity {

  private IjkVideoView videoView;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    videoView = (IjkVideoView) findViewById(R.id.video_view);
    videoView.setAspectRatio(IRenderView.AR_ASPECT_FIT_PARENT);
    videoView.setVideoURI(Uri.parse("http://zv.3gv.ifeng.com/live/zhongwen800k.m3u8"));
    videoView.start();
  }
}

常用函数

/**
 * 参数aspectRatio(缩放参数)参见IRenderView的常量:IRenderView.AR_ASPECT_FIT_PARENT,
IRenderView.AR_ASPECT_FILL_PARENT,
IRenderView.AR_ASPECT_WRAP_CONTENT,
IRenderView.AR_MATCH_PARENT,
IRenderView.AR_16_9_FIT_PARENT,
IRenderView.AR_4_3_FIT_PARENT
 */
public void setAspectRatio(int aspectRatio);
//改变视频缩放状态。
public int toggleAspectRatio();
//设置视频路径。
public void setVideoPath(String path);
//设置视频URI。(可以是网络视频地址)
public void setVideoURI(Uri uri);
//停止视频播放,并释放资源。
public void stopPlayback();
/**
 * 设置媒体控制器。
 * 参数controller:媒体控制器,注意是com.hx.ijkplayer_demo.widget.media.IMediaController。
 */
public void setMediaController(IMediaController controller);
//改变媒体控制器显隐
private void toggleMediaControlsVisiblity();
//注册一个回调函数,在视频预处理完成后调用。在视频预处理完成后被调用。此时视频的宽度、高度、宽高比信息已经获取到,此时可调用seekTo让视频从指定位置开始播放。
public void setOnPreparedListener(OnPreparedListener l);
//播放完成回调
public void setOnCompletionListener(IMediaPlayer.OnCompletionListener l);
//播放错误回调
public void setOnErrorListener(IMediaPlayer.OnErrorListener l);
//事件发生回调
public void setOnInfoListener(IMediaPlayer.OnInfoListener l);
//获取总长度
public int getDuration();
//获取当前播放位置。
public long getCurrentPosition();
//设置播放位置。单位毫秒
public void seekTo(long msec);
//是否正在播放。
public boolean isPlaying();
//获取缓冲百分比。
public int getBufferPercentage();

封装ijkplayer

上面只是ijkplayer的一个基本用法。我们可以对ijkplayer进行一次封装,让ijkplayer使用起来更加简单。

功能:

  • 使用Vitamio的VideoView进行视频播放
  • 视频左侧界面(左1/2以内)上下滑动调节亮度
  • 视频右侧界面(右1/2以外)上下滑动调节声音
  • 双击切换视频窗口布局
  • 非直播状态,可以左右滑动调节当前播放进度
public class PlayerManager {
  /**
   * 可能会剪裁,保持原视频的大小,显示在中心,当原视频的大小超过view的大小超过部分裁剪处理
   */
  public static final String SCALETYPE_FITPARENT="fitParent";
  /**
   * 可能会剪裁,等比例放大视频,直到填满View为止,超过View的部分作裁剪处理
   */
  public static final String SCALETYPE_FILLPARENT="fillParent";
  /**
   * 将视频的内容完整居中显示,如果视频大于view,则按比例缩视频直到完全显示在view中
   */
  public static final String SCALETYPE_WRAPCONTENT="wrapContent";
  /**
   * 不剪裁,非等比例拉伸画面填满整个View
   */
  public static final String SCALETYPE_FITXY="fitXY";
  /**
   * 不剪裁,非等比例拉伸画面到16:9,并完全显示在View中
   */
  public static final String SCALETYPE_16_9="16:9";
  /**
   * 不剪裁,非等比例拉伸画面到4:3,并完全显示在View中
   */
  public static final String SCALETYPE_4_3="4:3";

  /**
   * 状态常量
   */
  private final int STATUS_ERROR=-1;
  private final int STATUS_IDLE=0;
  private final int STATUS_LOADING=1;
  private final int STATUS_PLAYING=2;
  private final int STATUS_PAUSE=3;
  private final int STATUS_COMPLETED=4;

  private final Activity activity;
  private final IjkVideoView videoView;
  private final AudioManager audioManager;
  public GestureDetector gestureDetector;

  private boolean playerSupport;
  private boolean isLive = false;//是否为直播
  private boolean fullScreenOnly;
  private boolean portrait;

  private final int mMaxVolume;
  private int screenWidthPixels;
  private int currentPosition;
  private int status=STATUS_IDLE;
  private long pauseTime;
  private String url;

  private float brightness=-1;
  private int volume=-1;
  private long newPosition = -1;
  private long defaultRetryTime=5000;

  private OrientationEventListener orientationEventListener;
  private PlayerStateListener playerStateListener;

  public void setPlayerStateListener(PlayerStateListener playerStateListener) {
    this.playerStateListener = playerStateListener;
  }

  private OnErrorListener onErrorListener=new OnErrorListener() {
    @Override
    public void onError(int what, int extra) {
    }
  };

  private OnCompleteListener onCompleteListener=new OnCompleteListener() {
    @Override
    public void onComplete() {
    }
  };

  private OnInfoListener onInfoListener=new OnInfoListener(){
    @Override
    public void onInfo(int what, int extra) {

    }
  };
  private OnControlPanelVisibilityChangeListener onControlPanelVisibilityChangeListener=new OnControlPanelVisibilityChangeListener() {
    @Override
    public void change(boolean isShowing) {
    }
  };

  /**
   * try to play when error(only for live video)
   * @param defaultRetryTime millisecond,0 will stop retry,default is 5000 millisecond
   */
  public void setDefaultRetryTime(long defaultRetryTime) {
    this.defaultRetryTime = defaultRetryTime;
  }

  public PlayerManager(final Activity activity) {
    try {
      IjkMediaPlayer.loadLibrariesOnce(null);
      IjkMediaPlayer.native_profileBegin("libijkplayer.so");
      playerSupport=true;
    } catch (Throwable e) {
      Log.e("GiraffePlayer", "loadLibraries error", e);
    }
    this.activity=activity;
    screenWidthPixels = activity.getResources().getDisplayMetrics().widthPixels;

    videoView = (IjkVideoView) activity.findViewById(R.id.video_view);
    videoView.setOnCompletionListener(new IMediaPlayer.OnCompletionListener() {
      @Override
      public void onCompletion(IMediaPlayer mp) {
        statusChange(STATUS_COMPLETED);
        onCompleteListener.onComplete();
      }
    });
    videoView.setOnErrorListener(new IMediaPlayer.OnErrorListener() {
      @Override
      public boolean onError(IMediaPlayer mp, int what, int extra) {
        statusChange(STATUS_ERROR);
        onErrorListener.onError(what,extra);
        return true;
      }
    });
    videoView.setOnInfoListener(new IMediaPlayer.OnInfoListener() {
      @Override
      public boolean onInfo(IMediaPlayer mp, int what, int extra) {
        switch (what) {
          case IMediaPlayer.MEDIA_INFO_BUFFERING_START:
            statusChange(STATUS_LOADING);
            break;
          case IMediaPlayer.MEDIA_INFO_BUFFERING_END:
            statusChange(STATUS_PLAYING);
            break;
          case IMediaPlayer.MEDIA_INFO_NETWORK_BANDWIDTH:
            //显示下载速度
//           Toast.show("download rate:" + extra);
            break;
          case IMediaPlayer.MEDIA_INFO_VIDEO_RENDERING_START:
            statusChange(STATUS_PLAYING);
            break;
        }
        onInfoListener.onInfo(what,extra);
        return false;
      }
    });

    audioManager = (AudioManager) activity.getSystemService(Context.AUDIO_SERVICE);
    mMaxVolume = audioManager.getStreamMaxVolume(AudioManager.STREAM_MUSIC);
    gestureDetector = new GestureDetector(activity, new PlayerGestureListener());

    if (fullScreenOnly) {
      activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);
    }
    portrait=getScreenOrientation()== ActivityInfo.SCREEN_ORIENTATION_PORTRAIT;

    if (!playerSupport) {
      DebugLog.e("播放器不支持此设备");
    }
  }

  private void statusChange(int newStatus) {
    status = newStatus;
    if (!isLive && newStatus==STATUS_COMPLETED) {
      DebugLog.d("statusChange STATUS_COMPLETED...");
      if (playerStateListener != null){
        playerStateListener.onComplete();
      }
    }else if (newStatus == STATUS_ERROR) {
      DebugLog.d("statusChange STATUS_ERROR...");
      if (playerStateListener != null){
        playerStateListener.onError();
      }
    } else if(newStatus==STATUS_LOADING){
//      $.id(R.id.app_video_loading).visible();
      if (playerStateListener != null){
        playerStateListener.onLoading();
      }
      DebugLog.d("statusChange STATUS_LOADING...");
    } else if (newStatus == STATUS_PLAYING) {
      DebugLog.d("statusChange STATUS_PLAYING...");
      if (playerStateListener != null){
        playerStateListener.onPlay();
      }
    }
  }

  public void onPause() {
    pauseTime= System.currentTimeMillis();
    if (status==STATUS_PLAYING) {
      videoView.pause();
      if (!isLive) {
        currentPosition = videoView.getCurrentPosition();
      }
    }
  }

  public void onResume() {
    pauseTime=0;
    if (status==STATUS_PLAYING) {
      if (isLive) {
        videoView.seekTo(0);
      } else {
        if (currentPosition>0) {
          videoView.seekTo(currentPosition);
        }
      }
      videoView.start();
    }
  }

  public void onDestroy() {
    orientationEventListener.disable();
    videoView.stopPlayback();
  }

  public void play(String url) {
    this.url = url;
    if (playerSupport) {
      videoView.setVideoPath(url);
      videoView.start();
    }
  }

  private String generateTime(long time) {
    int totalSeconds = (int) (time / 1000);
    int seconds = totalSeconds % 60;
    int minutes = (totalSeconds / 60) % 60;
    int hours = totalSeconds / 3600;
    return hours > 0 ? String.format("%02d:%02d:%02d", hours, minutes, seconds) : String.format("%02d:%02d", minutes, seconds);
  }

  private int getScreenOrientation() {
    int rotation = activity.getWindowManager().getDefaultDisplay().getRotation();
    DisplayMetrics dm = new DisplayMetrics();
    activity.getWindowManager().getDefaultDisplay().getMetrics(dm);
    int width = dm.widthPixels;
    int height = dm.heightPixels;
    int orientation;
    // if the device's natural orientation is portrait:
    if ((rotation == Surface.ROTATION_0 || rotation == Surface.ROTATION_180) && height > width ||
        (rotation == Surface.ROTATION_90 || rotation == Surface.ROTATION_270) && width > height) {
      switch (rotation) {
        case Surface.ROTATION_0:
          orientation = ActivityInfo.SCREEN_ORIENTATION_PORTRAIT;
          break;
        case Surface.ROTATION_90:
          orientation = ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE;
          break;
        case Surface.ROTATION_180:
          orientation = ActivityInfo.SCREEN_ORIENTATION_REVERSE_PORTRAIT;
          break;
        case Surface.ROTATION_270:
          orientation = ActivityInfo.SCREEN_ORIENTATION_REVERSE_LANDSCAPE;
          break;
        default:
          orientation = ActivityInfo.SCREEN_ORIENTATION_PORTRAIT;
          break;
      }
    }
    // if the device's natural orientation is landscape or if the device
    // is square:
    else {
      switch (rotation) {
        case Surface.ROTATION_0:
          orientation = ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE;
          break;
        case Surface.ROTATION_90:
          orientation = ActivityInfo.SCREEN_ORIENTATION_PORTRAIT;
          break;
        case Surface.ROTATION_180:
          orientation = ActivityInfo.SCREEN_ORIENTATION_REVERSE_LANDSCAPE;
          break;
        case Surface.ROTATION_270:
          orientation = ActivityInfo.SCREEN_ORIENTATION_REVERSE_PORTRAIT;
          break;
        default:
          orientation = ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE;
          break;
      }
    }
    return orientation;
  }

  /**
   * 滑动改变声音大小
   *
   * @param percent
   */
  private void onVolumeSlide(float percent) {
    if (volume == -1) {
      volume = audioManager.getStreamVolume(AudioManager.STREAM_MUSIC);
      if (volume < 0)
        volume = 0;
    }
    int index = (int) (percent * mMaxVolume) + volume;
    if (index > mMaxVolume) {
      index = mMaxVolume;
    } else if (index < 0){
      index = 0;
    }
    // 变更声音
    audioManager.setStreamVolume(AudioManager.STREAM_MUSIC, index, 0);
    // 变更进度条
    int i = (int) (index * 1.0 / mMaxVolume * 100);
    String s = i + "%";
    if (i == 0) {
      s = "off";
    }
    DebugLog.d("onVolumeSlide:"+s);
  }

  private void onProgressSlide(float percent) {
    long position = videoView.getCurrentPosition();
    long duration = videoView.getDuration();
    long deltaMax = Math.min(100 * 1000, duration - position);
    long delta = (long) (deltaMax * percent);

    newPosition = delta + position;
    if (newPosition > duration) {
      newPosition = duration;
    } else if (newPosition <= 0) {
      newPosition=0;
      delta=-position;
    }
    int showDelta = (int) delta / 1000;
    if (showDelta != 0) {
      String text = showDelta > 0 ? ("+" + showDelta) : "" + showDelta;
      DebugLog.d("onProgressSlide:" + text);
    }
  }

  /**
   * 滑动改变亮度
   *
   * @param percent
   */
  private void onBrightnessSlide(float percent) {
    if (brightness < 0) {
      brightness = activity.getWindow().getAttributes().screenBrightness;
      if (brightness <= 0.00f){
        brightness = 0.50f;
      }else if (brightness < 0.01f){
        brightness = 0.01f;
      }
    }
    DebugLog.d("brightness:"+brightness+",percent:"+ percent);
    WindowManager.LayoutParams lpa = activity.getWindow().getAttributes();
    lpa.screenBrightness = brightness + percent;
    if (lpa.screenBrightness > 1.0f){
      lpa.screenBrightness = 1.0f;
    }else if (lpa.screenBrightness < 0.01f){
      lpa.screenBrightness = 0.01f;
    }
    activity.getWindow().setAttributes(lpa);
  }

  public void setFullScreenOnly(boolean fullScreenOnly) {
    this.fullScreenOnly = fullScreenOnly;
    tryFullScreen(fullScreenOnly);
    if (fullScreenOnly) {
      activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);
    } else {
      activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_SENSOR);
    }
  }

  private void tryFullScreen(boolean fullScreen) {
    if (activity instanceof AppCompatActivity) {
      ActionBar supportActionBar = ((AppCompatActivity) activity).getSupportActionBar();
      if (supportActionBar != null) {
        if (fullScreen) {
          supportActionBar.hide();
        } else {
          supportActionBar.show();
        }
      }
    }
    setFullScreen(fullScreen);
  }

  private void setFullScreen(boolean fullScreen) {
    if (activity != null) {
      WindowManager.LayoutParams attrs = activity.getWindow().getAttributes();
      if (fullScreen) {
        attrs.flags |= WindowManager.LayoutParams.FLAG_FULLSCREEN;
        activity.getWindow().setAttributes(attrs);
        activity.getWindow().addFlags(WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS);
      } else {
        attrs.flags &= (~WindowManager.LayoutParams.FLAG_FULLSCREEN);
        activity.getWindow().setAttributes(attrs);
        activity.getWindow().clearFlags(WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS);
      }
    }
  }

  /**
   * <pre>
   *   fitParent:可能会剪裁,保持原视频的大小,显示在中心,当原视频的大小超过view的大小超过部分裁剪处理
   *   fillParent:可能会剪裁,等比例放大视频,直到填满View为止,超过View的部分作裁剪处理
   *   wrapContent:将视频的内容完整居中显示,如果视频大于view,则按比例缩视频直到完全显示在view中
   *   fitXY:不剪裁,非等比例拉伸画面填满整个View
   *   16:9:不剪裁,非等比例拉伸画面到16:9,并完全显示在View中
   *   4:3:不剪裁,非等比例拉伸画面到4:3,并完全显示在View中
   * </pre>
   * @param scaleType
   */
  public void setScaleType(String scaleType) {
    if (SCALETYPE_FITPARENT.equals(scaleType)) {
      videoView.setAspectRatio(IRenderView.AR_ASPECT_FIT_PARENT);
    }else if (SCALETYPE_FILLPARENT.equals(scaleType)) {
      videoView.setAspectRatio(IRenderView.AR_ASPECT_FILL_PARENT);
    }else if (SCALETYPE_WRAPCONTENT.equals(scaleType)) {
      videoView.setAspectRatio(IRenderView.AR_ASPECT_WRAP_CONTENT);
    }else if (SCALETYPE_FITXY.equals(scaleType)) {
      videoView.setAspectRatio(IRenderView.AR_MATCH_PARENT);
    }else if (SCALETYPE_16_9.equals(scaleType)) {
      videoView.setAspectRatio(IRenderView.AR_16_9_FIT_PARENT);
    }else if (SCALETYPE_4_3.equals(scaleType)) {
      videoView.setAspectRatio(IRenderView.AR_4_3_FIT_PARENT);
    }
  }

  public void start() {
    videoView.start();
  }

  public void pause() {
    videoView.pause();
  }

  public boolean onBackPressed() {
    if (!fullScreenOnly && getScreenOrientation() == ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE) {
      activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT);
      return true;
    }
    return false;
  }

  class Query {
    private final Activity activity;
    private View view;

    public Query(Activity activity) {
      this.activity=activity;
    }

    public Query id(int id) {
      view = activity.findViewById(id);
      return this;
    }

    public Query image(int resId) {
      if (view instanceof ImageView) {
        ((ImageView) view).setImageResource(resId);
      }
      return this;
    }

    public Query visible() {
      if (view != null) {
        view.setVisibility(View.VISIBLE);
      }
      return this;
    }

    public Query gone() {
      if (view != null) {
        view.setVisibility(View.GONE);
      }
      return this;
    }

    public Query invisible() {
      if (view != null) {
        view.setVisibility(View.INVISIBLE);
      }
      return this;
    }

    public Query clicked(View.OnClickListener handler) {
      if (view != null) {
        view.setOnClickListener(handler);
      }
      return this;
    }

    public Query text(CharSequence text) {
      if (view!=null && view instanceof TextView) {
        ((TextView) view).setText(text);
      }
      return this;
    }

    public Query visibility(int visible) {
      if (view != null) {
        view.setVisibility(visible);
      }
      return this;
    }

    private void size(boolean width, int n, boolean dip){
      if(view != null){
        ViewGroup.LayoutParams lp = view.getLayoutParams();
        if(n > 0 && dip){
          n = dip2pixel(activity, n);
        }
        if(width){
          lp.width = n;
        }else{
          lp.height = n;
        }
        view.setLayoutParams(lp);
      }
    }

    public void height(int height, boolean dip) {
      size(false,height,dip);
    }

    public int dip2pixel(Context context, float n){
      int value = (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, n, context.getResources().getDisplayMetrics());
      return value;
    }

    public float pixel2dip(Context context, float n){
      Resources resources = context.getResources();
      DisplayMetrics metrics = resources.getDisplayMetrics();
      float dp = n / (metrics.densityDpi / 160f);
      return dp;
    }
  }

  public class PlayerGestureListener extends GestureDetector.SimpleOnGestureListener {
    private boolean firstTouch;
    private boolean volumeControl;
    private boolean toSeek;

    /**
     * 双击
     */
    @Override
    public boolean onDoubleTap(MotionEvent e) {
      videoView.toggleAspectRatio();
      return true;
    }

    @Override
    public boolean onDown(MotionEvent e) {
      firstTouch = true;
      return super.onDown(e);
    }

    /**
     * 滑动
     */
    @Override
    public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) {
      float mOldX = e1.getX(), mOldY = e1.getY();
      float deltaY = mOldY - e2.getY();
      float deltaX = mOldX - e2.getX();
      if (firstTouch) {
        toSeek = Math.abs(distanceX) >= Math.abs(distanceY);
        volumeControl=mOldX > screenWidthPixels * 0.5f;
        firstTouch = false;
      }

      if (toSeek) {
        if (!isLive) {
          onProgressSlide(-deltaX / videoView.getWidth());
        }
      } else {
        float percent = deltaY / videoView.getHeight();
        if (volumeControl) {
          onVolumeSlide(percent);
        } else {
          onBrightnessSlide(percent);
        }
      }

      return super.onScroll(e1, e2, distanceX, distanceY);
    }

    @Override
    public boolean onSingleTapUp(MotionEvent e) {
      return true;
    }
  }

  /**
   * is player support this device
   * @return
   */
  public boolean isPlayerSupport() {
    return playerSupport;
  }

  /**
   * 是否正在播放
   * @return
   */
  public boolean isPlaying() {
    return videoView!=null?videoView.isPlaying():false;
  }

  public void stop(){
    videoView.stopPlayback();
  }

  public int getCurrentPosition(){
    return videoView.getCurrentPosition();
  }

  /**
   * get video duration
   * @return
   */
  public int getDuration(){
    return videoView.getDuration();
  }

  public PlayerManager playInFullScreen(boolean fullScreen){
    if (fullScreen) {
      activity.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);
    }
    return this;
  }

  public PlayerManager onError(OnErrorListener onErrorListener) {
    this.onErrorListener = onErrorListener;
    return this;
  }

  public PlayerManager onComplete(OnCompleteListener onCompleteListener) {
    this.onCompleteListener = onCompleteListener;
    return this;
  }

  public PlayerManager onInfo(OnInfoListener onInfoListener) {
    this.onInfoListener = onInfoListener;
    return this;
  }

  public PlayerManager onControlPanelVisibilityChange(OnControlPanelVisibilityChangeListener listener){
    this.onControlPanelVisibilityChangeListener = listener;
    return this;
  }

  /**
   * set is live (can't seek forward)
   * @param isLive
   * @return
   */
  public PlayerManager live(boolean isLive) {
    this.isLive = isLive;
    return this;
  }

  public PlayerManager toggleAspectRatio(){
    if (videoView != null) {
      videoView.toggleAspectRatio();
    }
    return this;
  }

  public interface PlayerStateListener{
    void onComplete();
    void onError();
    void onLoading();
    void onPlay();
  }

  public interface OnErrorListener{
    void onError(int what, int extra);
  }

  public interface OnCompleteListener{
    void onComplete();
  }

  public interface OnControlPanelVisibilityChangeListener{
    void change(boolean isShowing);
  }

  public interface OnInfoListener{
    void onInfo(int what, int extra);
  }
}

使用封装后的PlayerManager播放视频:

public class MainActivity extends AppCompatActivity implements PlayerManager.PlayerStateListener{
  private String url1 = "rtmp://203.207.99.19:1935/live/CCTV5";
  private String url2 = "http://zv.3gv.ifeng.com/live/zhongwen800k.m3u8";
  private String url3 = "rtsp://184.72.239.149/vod/mp4:BigBuckBunny_115k.mov";
  private String url4 = "http://42.96.249.166/live/24035.m3u8";
  private PlayerManager player;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    initPlayer();
  }

  private void initPlayer() {
    player = new PlayerManager(this);
    player.setFullScreenOnly(true);
    player.setScaleType(PlayerManager.SCALETYPE_FILLPARENT);
    player.playInFullScreen(true);
    player.setPlayerStateListener(this);
    player.play(url1);
  }

  @Override
  public boolean onTouchEvent(MotionEvent event) {
    if (player.gestureDetector.onTouchEvent(event))
      return true;
    return super.onTouchEvent(event);
  }

  @Override
  public void onComplete() {
  }

  @Override
  public void onError() {
  }

  @Override
  public void onLoading() {
  }

  @Override
  public void onPlay() {
  }
}

Demo下载地址

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android:Field can be converted to a local varible.的解决办法

    Android:Field can be converted to a local varible.的解决办法 前言: 使用 Android Studio 开发 Android 有一段时间了,偶尔会碰到 AS 在一些私有变量上有黄色高亮提示Field can be converted to a local varible,有些强迫症的我还是不希望看到这个黄色的高亮.百度没查到什么有用的信息,还是用谷歌搜到了一些解答. 解析 Field can be converted to a local va

  • Android开发中使用WebView控件浏览网页的方法详解

    本文实例讲述了Android开发中使用WebView控件浏览网页的方法.分享给大家供大家参考,具体如下: 项目中遇到数学展示问题,常规的Textview显示处理不了数学公式,利用图片生成对服务器又产生较大压力,经过查询,可以通过webview加载JS实现.IOS同样的方法也可实现,但JS渲染效率远高于安卓.对Webview做下总结. 1.WebView 在使用WebView控件时,首先需要在xml布局文件中定义一个WebView控件,定义的方法如下: <WebView android:id=&quo

  • Android Studio时间选择器的创建方法

    本文实例为大家分享了Android九宫格图片展示的具体代码,供大家参考,具体内容如下 效果显示: 1.创建xml页面(我的项目扣下来的,有的地方会报错要改) <TextView android:id="@+id/consultation_tv_birthdate" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_a

  • Android使用RSA加密和解密的示例代码

    一.公钥加密和私钥解密 /**RSA算法*/ public static final String RSA = "RSA"; /**加密方式,android的*/ // public static final String TRANSFORMATION = "RSA/None/NoPadding"; /**加密方式,标准jdk的*/ public static final String TRANSFORMATION = "RSA/None/PKCS1Pad

  • Android编程监听APK安装与删除等过程的方法

    本文实例讲述了Android编程监听APK安装与删除等过程的方法.分享给大家供大家参考,具体如下: 软件下载后的一系列动作监听:先前是通过Service监听扫描获取状态,以后用这个方法测试使用 import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.widget.Toast; public class getBro

  • Android编程实现获取当前系统语言及地区并更改语言的方法

    本文实例讲述了Android编程实现获取当前系统语言及地区并更改语言的方法.分享给大家供大家参考,具体如下: 如果想获取手机的当前系统语言,可以通过Locale类获取,主要方法: Locale.getDefault().getLanguage() 返回的是es或者zh:通过 Locale.getDefault().getCountry() 获取当前国家或地区,返回为CN或US: 如果当前手机设置为中文-中国,则使用此方法返回zh-CN,同理可得到其他语言与地区的信息. 1.首先,如果要程序自动适

  • Android Studio实现带边框的圆形头像

    本文实例为大家分享了Android Studio实现带边框的圆形头像的具体代码,供大家参考,具体内容如下 效果显示: (没有边框的) (有边框的) 1.创建自定义ImagView控件 (1).没有边框的 package chenglong.activitytest.pengintohospital.utils; import android.content.Context; import android.graphics.Bitmap; import android.graphics.Bitma

  • Android实现单页面浮层可拖动view的一种方法

    上一篇讲到通过通过goolge官方的ViewDragHelper工具实现拖动的方法(上一篇见http://www.jb51.net/article/125481.htm),那么有一个问题就是在DragframeLayout中的onTouchEvent一直接收不到触摸消息,而且在onInterceptTouchEvent的时候,并没有触发ViewDragHelper.tryCaptureView方法,因此诞生了另一种比较原始的方法:通过自定义可拖动view来实现 主要方法: initEdge:设置

  • Android ijkplayer的使用方法解析

    ijkplayer是Bilibili基于ffmpeg开发并开源的轻量级视频播放器,支持播放本地网络视频,也支持流媒体播放.支持Android&iOS. ijkplayer的编译这里不多阐述,我也是直接获取别人编译完成的so库文件,直接使用的.如果你对ijkplayer的编译感兴趣,可以百度一下,有很多文章. 使用ijkplayer 导包 ijkplayer源码官方下载地址:https://github.com/Bilibili/ijkplayer 上面是官方提供的ijkplayer的源码地址,但

  • android textview 显示html方法解析

    现在网络的繁盛时代,光文字是不能满足人们的胃口的,图片,flash,音频,视频就成为浏览网页的主流显示,在手机上也一样.在手机上显示从网络端获取的数据显示,大家很自然的想起两种方式,一种就是webview,一种就是TextView.当然webView直接显示html页面就行了,我主要说的TextView显示html内容. 首先,说下TextView到底支持那些标签呢,通过对源码的查看,发现Textview可以解析一部分html标签,如: 复制代码 代码如下: <a href="...&qu

  • android实现图片验证码方法解析(自绘控件)

    自绘控件的内容都是自己绘制出来的 大致流程如下: 1.定义一个类继承view 1.使用TypedArray初始化属性集合     在view的构造方法中 有一个AttributeSet的参数 很明显是用来保存控件属性信息的 我们也的确可以通过循环然后用键值对的方式获取信息 而TypedArray是用来简化我们的工作的 2.重写onMeasure 测量控件大小 3.重写onDraw 绘制控件 2.根据需求在attrs文件中自定义属性 declare-styleable 声明自定义属性可以自定义一个

  • Android中gson、jsonobject解析JSON的方法详解

    JSON的定义: 一种轻量级的数据交换格式,具有良好的可读和便于快速编写的特性.业内主流技术为其提供了完整的解决方案(有点类似于正则表达式 ,获得了当今大部分语言的支持),从而可以在不同平台间进行数据交换.JSON采用兼容性很高的文本格式,同时也具备类似于C语言体系的行为. JSON对象: JSON中对象(Object)以"{"开始, 以"}"结束. 对象中的每一个item都是一个key-value对, 表现为"key:value"的形式, ke

  • Android编程使用pull方式解析xml格式文件的方法详解

    本文实例讲述了Android编程使用pull方式解析xml格式文件的方法.分享给大家供大家参考,具体如下: 上次已经说过使用Android sax解析xml,实际上还可以使用pull解析xml.这样的方式效率也是比较高的.pull不仅可以在Android上使用也可以用在javaee里面,需要的就是pull的jar包.这次的xml也使用上次的那个,如下所示 <?xml version="1.0" encoding="UTF-8"?> <persons

  • android编程之XML文件解析方法详解(附源码)

    本文实例讲述了android编程之XML文件解析方法.分享给大家供大家参考,具体如下: 在android开发中,经常用到去解析xml文件,常见的解析xml的方式有一下三种:SAX.Pull.Dom解析方式.最近做了一个android版的CSDN阅读器,用到了其中的两种(sax,pull),今天对android解析xml的这三种方式进行一次总结. 今天解析的xml示例(channels.xml)如下: <?xml version="1.0" encoding="utf-8

  • Android 使用Pull方法解析XML文件的方法

    Pull解析方法给应用程序完全的控制文档该怎么样被解析.Android中对Pull方法提供了支持的API,主要是 复制代码 代码如下: org.xmlpull.v1.XmlPullParser;org.xmlpull.v1.XmlPullParserFactory; 二个类,其中主要使用的是XmlPullParser,XmlPullParserFactory是一个工厂,用于构建XmlPullParser对象.应用程序通过调用XmlPullParser.next()等方法来产生Event,然后再处

  • Android编程简易实现XML解析的方法详解

    本文实例讲述了Android编程简易实现XML解析的方法.分享给大家供大家参考,具体如下: 首先创建在Android工程中创建一个Assets文件夹 app/src/main/assets 在这里添加一个名为 data.xml的文件,然后编辑这个文件,加入如下XML格式内容 <?xml version="1.0" encoding="utf-8"?> <apps> <app> <id>1</id> <

  • Android ListView 单条刷新方法实践及原理解析

    对于使用listView配合adapter进行刷新的方法大家都不陌生,先刷新adapter里的数据,然后调用notifydatasetchange通知listView刷新界面. 方法虽然简单,但这里面涉及到一个效率的问题,调用notifydatasetchange其实会导致adpter的getView方法被多次调用 (画面上能显示多少就会被调用多少次),如果是很明确的知道只更新了list中的某一个项的数据(比如用户点击list某一项后更新该项的显示状态,或者 后台回调更新list某一项,等等),

  • Android scrollTo和scrollBy方法使用解析

    在一个View中,系统提供了scrollTo.scrollBy两种方式来改变一个View的位置.这两个方法的区别非常好理解,与英文中To与By的区别类似,scrollTo(x, y)标识移动到一个具体的坐标点(x, y),而scrollBy(dx, dy)表示移动的增量为dx.dy. 在获取偏移量后使用scrollBy来移动View,代码如下所示: int offsetX = x - lastX; int offsetY = y - lastY; scrollBy(offsetX, offset

随机推荐