利用SurfaceView实现视频的播放
之前还有一篇文章是利用VideoView实现的视频播放,具体见: 视频播放-VideoView 并且这个系统封装好了,播放的时候不容易出错
代码如下: 关于重置的使用见: 音频播放
import android.media.AudioManager; import android.media.MediaPlayer; import android.os.Bundle; import android.os.Handler; import android.os.Message; import android.support.v7.app.AppCompatActivity; import android.util.Log; import android.view.SurfaceHolder; import android.view.SurfaceView; import android.view.View; import android.widget.ImageView; import android.widget.SeekBar; import android.widget.TextView; /** * 表面视图SurfaceView的使用,播放视频,SurfaceView只是用来做展示的 */ public class MainActivity extends AppCompatActivity implements View.OnClickListener, SeekBar.OnSeekBarChangeListener { /** * 展示画面 */ private SurfaceView surfaceView; /** * 媒体播放,播放声音 */ private MediaPlayer player; /** * 播放进度的显示 */ private SeekBar seekBar; /** * 显示播放的时间 */ private TextView textView; /** * 暂停与开始播放的按钮 */ private ImageView imageView; /** * 控制循环的标志位 */ private boolean flag = true; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); seekBar = (SeekBar) findViewById(R.id.seekBar); textView = (TextView) findViewById(R.id.textView); imageView = (ImageView) findViewById(R.id.imageView); imageView.setOnClickListener(this); seekBar.setOnSeekBarChangeListener(this); //找到表面视图 surfaceView = (SurfaceView) findViewById(R.id.surfaceView); SurfaceHolder holder = surfaceView.getHolder(); holder.addCallback(new SurfaceHolder.Callback() { @Override public void surfaceCreated(SurfaceHolder holder) { Log.i("tag", "主线程运行的,创建表面视图"); //必须在surface创建后才能初始化MediaPlayer,否则不会显示图像 player=MediaPlayer.create(MainActivity.this,R.raw.hello); player.setAudioStreamType(AudioManager.STREAM_MUSIC); //设置视频显示在SurfaceView上 player.setDisplay(holder); //player.setDataSource("路径或者网址");指定资源的第二种方式 //player.prepare();//这是同步的准备,如果是在创建媒体对象的时候就指定了播放源,就不用写这句了,否则异常 //player.prepareAsync();//这是异步的准备,因为是获取网络上的音频,如果用同步会出现页面卡顿的现象 //准备完毕的监听事件 player.setOnPreparedListener(new MediaPlayer.OnPreparedListener() { @Override public void onPrepared(MediaPlayer mp) { Log.i("tag", "播放的准备工作完毕"); seekBar.setMax(player.getDuration());//设置滑动条的最大值为视频的总时间 handler.sendEmptyMessageDelayed(1,1000);//发送消息,实现循环 } }); //音乐播放结束的监听 player.setOnCompletionListener(new MediaPlayer.OnCompletionListener() { @Override public void onCompletion(MediaPlayer mp) { Log.i("tag", "播放完毕"); flag = false;//结束handler消息的发送,这里可以将player重置下reset } }); } @Override public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) { Log.i("tag", "主线程中运行的,改变了,第一次创建也会调用"); } @Override public void surfaceDestroyed(SurfaceHolder holder) { Log.i("tag", "主线程中运行的,销毁了,activity从可见变为不可见就会调用"); if(player.isPlaying()){ player.stop(); } player.release();//释放资源 player = null; } }); } @Override public void onClick(View v) { if(player.isPlaying()){//正在播放 player.pause();//暂停, 设置为暂停的图标 imageView.setImageResource(android.R.drawable.ic_media_play); }else{ player.start();//开始或者继续播放 imageView.setImageResource(android.R.drawable.ic_media_pause); } } /** * seekBar的监听事件 */ @Override public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) { } @Override public void onStartTrackingTouch(SeekBar seekBar) { player.pause();//开始拖动时,暂停播放 } @Override public void onStopTrackingTouch(SeekBar seekBar) { player.seekTo(seekBar.getProgress());//播放到滑动条停止滑动的位置 player.start();//停止拖动后,继续播放 } /** * 更新显示 */ private Handler handler = new Handler(){ @Override public void handleMessage(Message msg) { if(msg.what == 1){ updataTime();//类似于递归,调用方法向自己发送消息 } } }; /** * 如果是开线程实现更新数据,在关闭的时候容易出现关不了的情况,前面有利用SufaceView实现gif图片的播放,摄像头画面预览就出现了这种情况 * 利用handler向自己发送消息,实现循环,这样能很好的避免线程关闭不了的情况了 */ private void updataTime(){ if(flag && player != null){ int currentPosition = player.getCurrentPosition();//当前播放的毫秒数 seekBar.setProgress(currentPosition); textView.setText(simpleTime(currentPosition)+"/"+simpleTime(seekBar.getMax())); handler.sendEmptyMessageDelayed(1,1000); } } /** * 格式化时间,2d代表二个十进制数,02d代表当不足两位时前面补零 */ private String simpleTime(int time){ int hour = time / 1000 / 3600; int minute = time / 1000 % 3600 /60; int second = time / 1000 % 60; return String.format("%02d:%02d:%02d",hour,minute,second);//00:00:00的样式 } @Override protected void onDestroy() { super.onDestroy(); } }
布局文件如下:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout 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" android:orientation="vertical"> <!--播放视频--> <SurfaceView android:layout_width="match_parent" android:layout_height="300dp" android:id="@+id/surfaceView" /> <LinearLayout android:orientation="horizontal" android:layout_width="match_parent" android:layout_height="wrap_content"> <ImageView android:id="@+id/imageView" android:src="@android:drawable/ic_media_play" android:layout_width="wrap_content" android:layout_height="wrap_content" /> <SeekBar android:id="@+id/seekBar" android:layout_gravity="center_vertical" android:layout_width="0dp" android:layout_weight="1" android:layout_height="wrap_content" /> <TextView android:id="@+id/textView" android:text="00:00:00/00:00:00" android:textSize="20sp" android:layout_gravity="center_vertical" android:layout_width="wrap_content" android:layout_height="wrap_content" /> </LinearLayout> </LinearLayout>
效果如下: