44_在线视频播放器
-------------------------
1.注意这里,在模拟器中,android2.2和android2.1视频是没有播放效果的,这个主要是因为模拟器的原因和程序代码无关
------------------------------------------------------------------------------------------------------------
2.这里可以使用android2.0的平台进行测试播放.是可以成功的.
-----------------------------------------------------------
3.在main.xml布局文件添加用于视频画面绘制的SurfaceView 控件:
<SurfaceView android:layout_width="fill_parent" android:layout_height="240dip" android:id="@+id/surfaceView" />
SurfaceView surfaceView = (SurfaceView)this.findViewById(R.id.surfaceView);
surfaceView.getHolder().setFixedSize(176, 144); //设置分辨率
/*下面设置Surface不维护自己的缓冲区,而是等待屏幕的渲染引擎将内容推送到用户面前*/
surfaceView.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
MediaPlayer mediaPlayer = new MediaPlayer();
mediaPlayer.reset();//重置为初始状态
mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
/* 设置Video影片以SurfaceHolder播放 */
mediaPlayer.setDisplay(surfaceView.getHolder());
mediaPlayer.setDataSource("/mnt/sdcard/oppo.mp4");
mediaPlayer.prepare();//缓冲
mediaPlayer.start();//播放
mediaPlayer.pause();//暂停播放
mediaPlayer.start();//恢复播放
mediaPlayer.stop();//停止播放
mediaPlayer.release();//释放资源
------------------------------------------------
4.package cn.itcast.video;
import java.io.IOException;
import android.app.Activity;
import android.media.AudioManager;
import android.media.MediaPlayer;
import android.os.Bundle;
import android.util.Log;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.View;
import android.widget.EditText;
import android.widget.ImageButton;
public class VideoActivity extends Activity {
private static final String TAG = "VideoActivity";
private EditText filenameText;
private SurfaceView surfaceView;
private MediaPlayer mediaPlayer;
private String filename;//当前播放文件的名称
private int position;//记录播放位置
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
this.mediaPlayer = new MediaPlayer();
this.filenameText = (EditText) this.findViewById(R.id.filename);
this.surfaceView = (SurfaceView) this.findViewById(R.id.surfaceView);
ImageButton playButton = (ImageButton) this.findViewById(R.id.play);
ImageButton pauseButton = (ImageButton) this.findViewById(R.id.pause);
ImageButton resetButton = (ImageButton) this.findViewById(R.id.reset);
ImageButton stopButton = (ImageButton) this.findViewById(R.id.stop);
ButtonClickListener listener = new ButtonClickListener();
playButton.setOnClickListener(listener);
pauseButton.setOnClickListener(listener);
resetButton.setOnClickListener(listener);
stopButton.setOnClickListener(listener);
/*下面设置Surface不维护自己的缓冲区,而是等待屏幕的渲染引擎将内容推送到用户面前*/
this.surfaceView.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
this.surfaceView.getHolder().setFixedSize(176, 144);//设置分辨率
this.surfaceView.getHolder().setKeepScreenOn(true);
this.surfaceView.getHolder().addCallback(new SurfaceListener());
}
private class ButtonClickListener implements View.OnClickListener{
@Override
public void onClick(View v) {
try {
switch (v.getId()) {
case R.id.play://来自播放按钮
filename = filenameText.getText().toString();
play();
break;
case R.id.pause://来自暂停按钮
if(mediaPlayer.isPlaying()){
mediaPlayer.pause();
}else{
mediaPlayer.start();
}
break;
case R.id.reset://来自重新播放按钮
if(!mediaPlayer.isPlaying()) play();
mediaPlayer.seekTo(0);
break;
case R.id.stop://来自停止按钮
if(mediaPlayer.isPlaying()) mediaPlayer.stop();
break;
}
} catch (Exception e) {
Log.e(TAG, e.toString());
}
}
}
/**
* 播放视频
*/
private void play() throws IOException {
mediaPlayer.reset();
mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
mediaPlayer.setDataSource("/mnt/sdcard/"+ filename);//设置需要播放的视频
mediaPlayer.setDisplay(surfaceView.getHolder());//把视频画面输出到SurfaceView
mediaPlayer.prepare();
mediaPlayer.start();
}
private class SurfaceListener implements SurfaceHolder.Callback{
@Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
}
@Override
public void surfaceCreated(SurfaceHolder holder) {//方法在onResume()后被调用
Log.i(TAG, "surfaceCreated()");
if(position>0 && filename!=null){
try {
play();
mediaPlayer.seekTo(position);
position = 0;
} catch (Exception e) {
Log.e(TAG, e.toString());
}
}
}
@Override
public void surfaceDestroyed(SurfaceHolder holder) {
Log.i(TAG, "surfaceDestroyed()");
}
}
@Override
protected void onPause() {//当其他Activity被打开,停止播放
if(mediaPlayer.isPlaying()){
position = mediaPlayer.getCurrentPosition();//得到播放位置
mediaPlayer.stop();
}
super.onPause();
}
@Override
protected void onDestroy() {
if(mediaPlayer.isPlaying()) mediaPlayer.stop();
mediaPlayer.release();
super.onDestroy();
}
}
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:background="#FFFFFF"
android:orientation="vertical"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
>
<TextView
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="@string/filename"
/>
<EditText
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="oppo.mp4"
android:id="@+id/filename"
/>
<LinearLayout
android:orientation="horizontal"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
>
<ImageButton
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/play"
android:id="@+id/play"
/>
<ImageButton
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/pause"
android:id="@+id/pause"
/>
<ImageButton
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/reset"
android:id="@+id/reset"
/>
<ImageButton
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/stop"
android:id="@+id/stop"
/>
</LinearLayout>
<SurfaceView
android:layout_width="fill_parent"
android:layout_height="240dip"
android:id="@+id/surfaceView"
/>
</LinearLayout>
----------------------------------------------------------------
1.下面是这个视频播放器的所有代码;
-----------------------------------
a.创建android项目:videoplayer
b./videoplayer/src/com/credream/video/VideoplayerActivity.java
package com.credream.video;
import java.io.File;
import android.app.Activity;
import android.media.MediaPlayer;
import android.media.MediaPlayer.OnPreparedListener;
import android.os.Bundle;
import android.os.Environment;
import android.view.SurfaceHolder;
import android.view.SurfaceHolder.Callback;
import android.view.SurfaceView;
import android.view.View;
import android.widget.EditText;
import android.widget.Toast;
public class VideoplayerActivity extends Activity {
private EditText nameText;
private String path;
private MediaPlayer mediaPlayer;
private SurfaceView surfaceView;
private boolean pause;
private int position;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
mediaPlayer = new MediaPlayer();
/*取得文件名称*/
nameText = (EditText) this.findViewById(R.id.filename);
surfaceView = (SurfaceView) this.findViewById(R.id.surfaceView);
//把输送给surfaceView的视频画面,直接显示到屏幕上,不要维持它自身的缓冲区
surfaceView.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
//取得holder对象,设置控制类型,SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS,这个类型的作用:
//把输送给surfaceView的视频画面,直接显示到屏幕上,不要维持它自身的缓冲区
surfaceView.getHolder().setFixedSize(176, 144);
//设置显示视频的分辨率
surfaceView.getHolder().setKeepScreenOn(true);
//让屏幕不要暗下去,锁屏,也就是在播放的时候,屏幕一直亮着.
surfaceView.getHolder().addCallback(new SurfaceCallback());
//这里通过surfaceView.getHolder().addCallback这个方法就可以监听到surfaceView的创建状态
//当surfaceView被创建出来的时候就会调用SurfaceCallback类的surfaceCreated这个方法:
}
/*
* 当这个activity不在前台的时候调用这个方法onPause,停止视频播放
* 当这个activity回到前台的时候调用这个方法onResume,继续播放视频
* 但是这里出现了一个问题,在播放的时候,只有声音没有画面,这是由于surfaceView需要处理造成的
* 这个问题的原因:当surfaceView所在的activity离开了前台surfaceView会被摧毁会被destory
* 当activity又重新回到前台时候,surfaceView会被重新创建,surfaceView是在onResume()方法之后被
* 创建,由于surfaceView是在onResume方法之后创建的所以没有画面只有声音.
*
* */
// SurfaceCallback通过这个类实现
private final class SurfaceCallback implements Callback{
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
}//surfaceChanged这个方法是在绘图区域发生变化的时候自动调用这个方法
//当被创建的时候如果存在播放点就进入到播放点进行播放
public void surfaceCreated(SurfaceHolder holder) {
if(position>0 && path!=null){
play(position);
position = 0;
}
}
//当surfaceView所在的activity离开了前台surfaceView会被摧毁会被destory
//这时候先要记录下播放位置,在创建的 时候surfaceView继续播放
public void surfaceDestroyed(SurfaceHolder holder) {
if(mediaPlayer.isPlaying()){
position = mediaPlayer.getCurrentPosition();
mediaPlayer.stop();
}
}
}
//只要activity不在前台就会调用这个方法
//如果只希望当来电话的时候,停止播放,挂电话的时候继续播放,这时候就不可以用这两个方法,因为只要这个activity不在前台
//那么就会选择性的调用这两个方法.即:当activity回到前台时候调用onResume,开始播放,当activity回到后台的时候调用onPause停止播放
@Override
protected void onPause() {//停止
if(mediaPlayer.isPlaying()) {
position = mediaPlayer.getCurrentPosition();
mediaPlayer.stop();
}
super.onPause();
}
//当activity重新回到前台的时候,这个方法就会必然调用,继续播放
//如果只希望当来电话的时候,停止播放,挂电话的时候继续播放,这时候就不可以用这两个方法,因为只要这个activity不在前台
//那么就会选择性的调用这两个方法.即:当activity回到前台时候调用onResume,开始播放,当activity回到后台的时候调用onPause停止播放
@Override
protected void onResume() {
if(position>0 && path!=null){
play(position);
position = 0;
}
super.onResume();
}
@Override
protected void onDestroy() {
mediaPlayer.release();
mediaPlayer = null;
super.onDestroy();
}
/*按钮响应的方法,这个方法不用添加按钮监听,只要在配置文件中配置好就可以了*/
public void mediaplay(View v){
switch (v.getId()) {
case R.id.playbutton:
/**
* 如果需要播放网络上的视频,就需要在清单文件中加上网络访问权限.
* 另外直接这样播放网络上的视频这样是不行的,因为这个文件在网络上
* 还不是流媒体文件,是不可以直接播放的
* 下面是解决方法:
* 流媒体的分发方式,渐进式下载(Progressive Download)和实时流媒体
* 渐进式下载(Progressive Download)可以通过HTTP或FTP协议来分发,需要web服务器或者是Ftp服务器
* 实时流媒体通过RTP和RTSP这种实时协议来分发,需要一个流媒体服务器.
* 注意,渐进式媒体文是和媒体的格式有关的,mp4不是渐进式的媒体文件.需要转换成其他格式的比如
* 3gp才可以..注意,这里在转换格式的时候一般都需要设置数据速率,这里的数据速率一般要设置的
* 和自己的网速差不多的时候最好了.太快或者太慢都不好,但是如果这里设置的速率超过网络的带宽就会
* 在android系统中播放不出来.也不要太低,太低了,视频的效果会很差建议在160---230之间
* 图像的大小设置,没有关系,因为可以放大缩小.转换完成后直接放到webroot的目录下然后在应用中输入网络地址
* 就可以直接播放了.
* --------------------------------------------
* 在以前的时候,是通过把一个大的视频切割成很多小的视频,然后把一个很小的视频先下载到本地然后播放
* 然后实现类似于视频在线播放的功能,其实也是本地转换.
* ----------------------------------------------
* 在显示企业中如果需要开发视频播放的时候,是不可能手动实现转码的,这时候可以借用quickTime提供的一套api来实现.
* -------------------------------------
*/
String filename = nameText.getText().toString();
//这个时候,是指需要播放网络视频
if(filename.startsWith("http")){
path = filename;
play(0);
}else{
//如果只需要播放本地文件的话,就直接写这个就可以了.
File file = new File(Environment.getExternalStorageDirectory(), filename);
if(file.exists()){
path = file.getAbsolutePath();
play(0);
}else{
path = null;
Toast.makeText(this, R.string.filenoexsit, 1).show();
}
}
break;
case R.id.pausebutton:
if(mediaPlayer.isPlaying()){
mediaPlayer.pause();
pause = true;
}else{
if(pause){
mediaPlayer.start();
pause = false;
}
}
break;
case R.id.resetbutton:
if(mediaPlayer.isPlaying()){
mediaPlayer.seekTo(0);
}else{
if(path!=null){
play(0);
}
}
break;
case R.id.stopbutton:
if(mediaPlayer.isPlaying()){
mediaPlayer.stop();
}
break;
}
}
private void play(int position) {
try {
mediaPlayer.reset();
//path在上面定义并赋值完后在这里使用
mediaPlayer.setDataSource(path);
//这里需要把设置后的surfaceView的控制对象Holder进来.
mediaPlayer.setDisplay(surfaceView.getHolder());
//这个方法用来设置往activity的哪个地方显示视频,根据surfacView的设置情况
mediaPlayer.prepare();//缓冲
mediaPlayer.setOnPreparedListener(new PrepareListener(position));
//mediaPlayer.setOnPreparedListener这个是用来监听缓冲是否完成的方法,缓冲结束后会自动的调用onPrepared这个方法
} catch (Exception e) {
e.printStackTrace();
}
}
//缓冲是否完成的监听类
private final class PrepareListener implements OnPreparedListener{
private int position;
public PrepareListener(int position) {
this.position = position;
}
public void onPrepared(MediaPlayer mp) {
mediaPlayer.start();//播放视频
if(position>0) mediaPlayer.seekTo(position);
}
}
}
----------------------------------------------------------------------
c./videoplayer/res/values/strings.xml
<?xml version="1.0" encoding="utf-8"?>
<resources>
<string name="hello">Hello World, VideoplayerActivity!</string>
<string name="app_name">视频播放器</string>
<string name="filename">视频文件名称</string>
<string name="filenoexsit">视频文件不存在</string>
</resources>
-----------------------------------------------
d./videoplayer/res/layout/main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:background="#FFFFFF"
><!--
这里设置这个的颜色主要是为了和SurfaceView控件的背景颜色形成对比,
因为SurfaceView的背景颜色(也就是播放区域的颜色是黑色的这样就和背景颜色
混在一起没法区分所以这里把整体的背景颜色设置成了白色,这样就可以知道
视频播放的位置在哪里了) -->
<TextView
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="@string/filename"
/>
<EditText
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:id="@+id/filename"
/>
<LinearLayout
android:orientation="horizontal"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
>
<!-- android:src="@drawable/play"这个用来取得/videoplayer/res/drawable-hdpi文件夹下的图片 -->
<ImageButton
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/play"
android:id="@+id/playbutton"
android:onClick="mediaplay"
/>
<ImageButton
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/pause"
android:id="@+id/pausebutton"
android:onClick="mediaplay"
/>
<ImageButton
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/reset"
android:id="@+id/resetbutton"
android:onClick="mediaplay"
/>
<ImageButton
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/stop"
android:id="@+id/stopbutton"
android:onClick="mediaplay"
/>
</LinearLayout>
<!-- 这个控件是用来绘图的 -->
<SurfaceView
android:layout_width="fill_parent"
android:layout_height="240dp"
android:id="@+id/surfaceView"
/>
</LinearLayout>
-----------------------------------------------------
d./videoplayer/AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.credream.video"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk android:minSdkVersion="4" />
<application
android:icon="@drawable/ic_launcher"
android:label="@string/app_name" >
<activity
android:label="@string/app_name"
android:name=".VideoplayerActivity" >
<intent-filter >
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
<!-- 访问internet权限 -->
<uses-permission android:name="android.permission.INTERNET"/>
</manifest>
--------------------------------------------------------------------------
e.对于测试流媒体用的web应用,只需要把需要播放的渐进式流媒体文件,放到webroot文件夹下就行了.
然后启动android应用,把流媒体的地址,填写到应用的地址栏中就可以播放视频了.
-------------------------------------------------------------------------------