【Android 多媒体开发】 MediaPlayer 网络视频播放器(二)

简介: 【Android 多媒体开发】 MediaPlayer 网络视频播放器(二)

4. MediaPlayer 播放





(1) 设置音量 和 播放载体



设置音量 :



mediaPlayer.setAudioStreamType(2);    /* 设置播放音量 */


设置播放载体 : 调用 setDisplay() 方法, 传入 SurfaceHolder 对象;


mediaPlayer.setDisplay(surface_holder);  /* 设置播放载体 */


(2) 设置各种监听器



设置错误监听器 : 如果出现错误, 会回调该监听器中的方法, 并提供错误码;



/* 设置 MediaPlayer 错误监听器, 如果出现错误就会回调该方法打印错误代码 */
    mediaPlayer.setOnErrorListener(new OnErrorListener() {
    @Override
    public boolean onError(MediaPlayer arg0, int what, int extra) {
      System.out.println("MediaPlayer 出现错误 what : " + what + " , extra : " + extra);
      return false;
    }
    });


设置缓冲进度监听器 : 缓冲有进展后, 回调该监听器中的方法, 传入缓冲的数据百分比;


/* 设置缓冲进度更新监听器 */
    mediaPlayer.setOnBufferingUpdateListener(new OnBufferingUpdateListener() {
    @Override
    public void onBufferingUpdate(MediaPlayer arg0, int percent) {
      /* 打印缓冲的百分比, 如果缓冲 */
      System.out.println("缓冲了的百分比 : " + percent + " %");
    }
    });


设置播放完毕监听器 : 播放完毕后会回调该监听器中的方法;


/* 设置播放完毕监听器 */
    mediaPlayer.setOnCompletionListener(new OnCompletionListener() {
    @Override
    public void onCompletion(MediaPlayer arg0) {
      System.out.println("播放完毕了");
      status.setText("播放完毕");
    }
    });


设置准备完毕回调监听器 : 准备完毕后会回调该方法;


/* 设置准备完毕监听器 */
    mediaPlayer.setOnPreparedListener(new OnPreparedListener() {
    @Override
    public void onPrepared(MediaPlayer arg0) {
      System.out.println("准备完毕");
      /* 设置播放状态 */
      status.setText("播放中");
    }
    });





二. 代码示例



1. 布局文件代码




<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/container"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:layout_margin="20dp"
    tools:context="cn.org.octopus.videodemo.HomeActivity"
    tools:ignore="MergeRootFrame" 
    android:orientation="vertical">
    <AutoCompleteTextView 
        android:id="@+id/url"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:singleLine="true"
        android:ellipsize="end"
        android:completionThreshold="1"
        android:text="http://daily3gp.com/vids/747.3gp"
        android:completionHint="选择下载的视频地址"/>
    <SurfaceView 
        android:id="@+id/surface_view"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="3"/>
    <TextView 
        android:id="@+id/status"
        android:layout_width="match_parent"
        android:gravity="center"
        android:layout_height="0dp"
        android:layout_weight="1"
        android:textSize="20dp"
        android:text="状态"/>
    <LinearLayout 
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="1"
        android:gravity="center_vertical"
        android:orientation="horizontal">
        <Button 
            android:id="@+id/play"
            android:layout_width="0dp" 
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:onClick="onClick"
            android:text="播放"/>
        <Button 
            android:id="@+id/pause"
            android:layout_width="0dp" 
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:onClick="onClick"
            android:text="暂停"/>
        <Button 
            android:id="@+id/reset"
            android:layout_width="0dp" 
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:onClick="onClick"
            android:text="重放"/>
        <Button 
            android:id="@+id/stop"
            android:layout_width="0dp" 
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:onClick="onClick"
            android:text="停止"/>
    </LinearLayout>
</LinearLayout>




2. AndroidManifest 代码




<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="cn.org.octopus.videodemo"
    android:versionCode="1"
    android:versionName="1.0" >
    <uses-sdk
        android:minSdkVersion="17"
        android:targetSdkVersion="19" />
    <uses-permission android:name="android.permission.INTERNET"/>
    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name="cn.org.octopus.videodemo.HomeActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>
</manifest>



3. Activity 代码




package cn.org.octopus.videodemo;
import java.io.IOException;
import android.R.anim;
import android.app.Activity;
import android.content.DialogInterface.OnClickListener;
import android.graphics.PixelFormat;
import android.media.MediaPlayer;
import android.media.MediaPlayer.OnBufferingUpdateListener;
import android.media.MediaPlayer.OnCompletionListener;
import android.media.MediaPlayer.OnErrorListener;
import android.media.MediaPlayer.OnPreparedListener;
import android.os.Bundle;
import android.support.v4.view.ViewPager;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.View;
import android.widget.ArrayAdapter;
import android.widget.AutoCompleteTextView;
import android.widget.Button;
import android.widget.TextView;
/** 
 * 
 * 
 * SurfaceHolder  
 *    1. 简介 : 是 Surface 的控制器, 用于控制 SurfaceView 绘图, 处理画布上的动画, 渲染效果, 大小等;
 *    2. 常用方法 : 
 *    -- abstract void addCallback(SurfaceHolder.Callback callback) : 添加一个 SurfaceHolder.Callback 接口对象, 监听 Surface 的开始结束绘制大小改变事件;
 *    -- abstract Canvas lockCanvas() : 锁定画布, 可以获得 Canvas 对象, 之后就可以在 Canvas 上绘图了;
 * 
 * SurfaceHolder.Callback接口 : 
 *    1. Surface 绘图边界 : 所有的绘图工作都在 Surface 创建之后才能进行, 在 Surface 销毁之前结束;
 *    2. Callback 接口对应的 Surface 边界 : surfaceCreated() 方法在开始绘制时回调, surfaceDestroyed() 在 Surface 销毁前回调;
 *    3. 该接口中的方法 : 
 *    -- surfaceChanged() : 在 Surface 大小改变时回调;
 *    -- surfaceCreated() : 在 Surface 创建时回调;
 *    -- surfaceDestroyed() : 在 Surface 销毁时回调;
 * 
 * @author octopus 
 *
 */
public class HomeActivity extends Activity implements SurfaceHolder.Callback {
  private AutoCompleteTextView url;       /* 地址输入框, 带自动提示功能 */
  private SurfaceView surface_view;       /* 播放视频载体 */
  private TextView status;          /* 显示播放状态 */
  private Button play;          /* 播放按钮 */
  private Button pause;          /* 咱提供按钮 */
  private Button reset;          /* 重放按钮 */
  private Button stop;          /* 停止按钮 */
  private MediaPlayer mediaPlayer;        /* 播放器 */
  private SurfaceHolder surface_holder;       /* Surface 控制器 */
  private boolean isStartPlaying;  /* 是否开始了播放 */
  @Override
  protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_home);
  initViews();
  initData();
  }
  /**
  * 初始化成员变量中的组件变量
  */
  private void initViews() {
  /* 通过 findViewById 获取相关方法 */
  url = (AutoCompleteTextView) findViewById(R.id.url);
  surface_view = (SurfaceView) findViewById(R.id.surface_view);
  status = (TextView) findViewById(R.id.status);
  play = (Button) findViewById(R.id.play);
  pause = (Button) findViewById(R.id.pause);
  reset = (Button) findViewById(R.id.reset);
  stop = (Button) findViewById(R.id.stop);
  /* 设置一个列表适配器 */
  String[] urls = {
    "http://daily3gp.com/vids/747.3gp",
    "http://daily3gp.com/vids/Funny%20women%20cannot%20understand.3gp",
    "http://k.youku.com/player/getFlvPath/sid/9409280845322127f6c57_00/st/flv/fileid/0300020100540024BC9E5C08BD8A98D8200E2B-7950-B9A5-8669-DC283BDCC077?K=3a58dc2cdcc532df261dddec&ctype=12&ev=1&oip=1931322792&token=5696&ep=eyaUE0uFVsYE4CDdij8bYHrkJ3IIXP4J9h%2BFg9JjALshTOi%2FmzqjtJTFS4xCHottelMPGJ%2F5qdDnH0JmYfdKrGgQrUfZPPro%2BPbq5dkgxpgDFG1FAc3Qs1SbRTn3",
    "http://k.youku.com/player/getFlvPath/sid/9409280845322127f6c57_00/st/flv/fileid/030002040053FFB59E433100422C39BAFA46CC-4DED-E928-87B8-91706CDB5FF2?K=645d8478a3aa59052411eb8a&ctype=12&ev=1&oip=1931322792&token=5696&ep=eyaUE0uFVsYE4CDdij8bYHrkJ3IIXP4J9h%2BFg9JmALshS57J6zvYspmzTf5CFv0bcFEFGZmA3aHjbDNnYfQ33BwQqkeqMfro%2BYLr5aRSw5AGFW1Ed7uhtlSbRTn3"
  };
  /* 创建数组适配器 */
  ArrayAdapter<String> adapter = new ArrayAdapter<String>(this, 
    android.R.layout.simple_dropdown_item_1line, urls);
  /* 将适配器设置给 AutoCompleteTextView 组件对象 */
  url.setAdapter(adapter);
  /* 使窗口支持透明度, 把当前 Activity 窗口设置成透明, 设置了该选项就可以使用 setAlpha 等函数设置窗口透明度 */
  getWindow().setFormat(PixelFormat.TRANSPARENT);
  }
  /**
  * 初始化相关数据变量
  */
  private void initData() {
  /* 获取并设置 SurfaceHolder 对象 */
  surface_holder = surface_view.getHolder();      /* 根据 SurfaceView 组件, 获取 SurfaceHolder 对象 */
  surface_holder.addCallback(this);         /* 为 SurfaceHolder 设置回调函数, 即 SurfaceHolder.Callback 子类对象 */
  surface_holder.setFixedSize(160, 128);        /* 设置视频大小比例 */
  surface_holder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);/* 设置视频类型 */
  }
  /**
  * 设置点击事件
  * @param view
  */
  public void onClick(View view) {
  int id = view.getId();
  switch (id) {
  case R.id.play:
    /* 播放视频直接从 AutoCompleteTextView 中获取字符串, 播放该 url 代表的网络视频 */
    playVideo(url.getText().toString());
    break;
  case R.id.pause:
    if(mediaPlayer != null){
    mediaPlayer.pause();
    status.setText("暂停");
    }
    break;
  case R.id.reset:
    if(mediaPlayer != null){
    mediaPlayer.seekTo(0);
    mediaPlayer.start();
    status.setText("播放中");
    }
    break;
  case R.id.stop:
    if(mediaPlayer != null){
    mediaPlayer.stop();
    mediaPlayer.release();
    isStartPlaying = false;
    status.setText("停止");
    }
    break;
  default:
    break;
  }
  }
  /**
  * 播放网络视频
  * a. 创建并配置 MediaPlayer 对象 (音量, SurfaceHolder)
  * b. 为 MediaPlayer 设置错误监听器, 缓冲进度监听器, 播放完毕监听器, 准备完毕监听器
  * c. 未 MediaPlayer 设置数据源
  * d. 调用 prepare() 进入 Prapared 状态
  * e. 调用 start() 进入 Started 状态
  * 
  * @param dataSource 播放视频的网络地址
  */
  private void playVideo(final String dataSource) {
  /* 点击播放有两种情况 
   * a. 第一次点击 : 需要初始化 MediaPlayer 对象, 设置监听器
   * b. 第二次点击 : 只需要 调用 mediaPlayer 的 start() 方法
   * 两种情况通过 isStartPlaying 点击时间判断 */
  if(isStartPlaying){        /* 如果已经开始了播放, 就直接开始播放 */
    mediaPlayer.start();
  }else{            /* 如果是第一次开始播放, 需要初始化 MediaPlayer 设置监听器等操作 */
    mediaPlayer = new MediaPlayer();    /* 创建 MediaPlayer 对象 */
    mediaPlayer.setAudioStreamType(2);    /* 设置播放音量 */
    mediaPlayer.setDisplay(surface_holder);  /* 设置播放载体 */ 
    /* 设置 MediaPlayer 错误监听器, 如果出现错误就会回调该方法打印错误代码 */
    mediaPlayer.setOnErrorListener(new OnErrorListener() {
    @Override
    public boolean onError(MediaPlayer arg0, int what, int extra) {
      System.out.println("MediaPlayer 出现错误 what : " + what + " , extra : " + extra);
      return false;
    }
    });
    /* 设置缓冲进度更新监听器 */
    mediaPlayer.setOnBufferingUpdateListener(new OnBufferingUpdateListener() {
    @Override
    public void onBufferingUpdate(MediaPlayer arg0, int percent) {
      /* 打印缓冲的百分比, 如果缓冲 */
      System.out.println("缓冲了的百分比 : " + percent + " %");
    }
    });
    /* 设置播放完毕监听器 */
    mediaPlayer.setOnCompletionListener(new OnCompletionListener() {
    @Override
    public void onCompletion(MediaPlayer arg0) {
      System.out.println("播放完毕了");
      status.setText("播放完毕");
    }
    });
    /* 设置准备完毕监听器 */
    mediaPlayer.setOnPreparedListener(new OnPreparedListener() {
    @Override
    public void onPrepared(MediaPlayer arg0) {
      System.out.println("准备完毕");
      /* 设置播放状态 */
      status.setText("播放中");
    }
    });
    new Thread(){
    public void run() {
      try {
      System.out.println("设置数据源");
      mediaPlayer.setDataSource(dataSource);
      mediaPlayer.prepare();
      /* 打印播放视频的时长 */
      System.out.println("视频播放长度 : " + mediaPlayer.getDuration());
      mediaPlayer.start();
      } catch (IllegalStateException e) {
      e.printStackTrace();
      } catch (IOException e) {
      e.printStackTrace();
      }
    };
    }.start();
    /* 设置 MediaPlayer 开始播放标识为 true */
    isStartPlaying = true;
    /* 设置播放状态 */
    status.setText("正在缓冲");
  }
  }
  /**
  * 在 Surface 大小发生改变的时候回调
  * 实现的 SurfaceHolder.Callback 接口方法
  */
  @Override
  public void surfaceChanged(SurfaceHolder arg0, int arg1, int arg2, int arg3) {
  System.out.println("SurfaceHolder.Callback.surfaceChanged : Surface 大小发生改变");
  }
  /**
  * 在 Surface 创建的时候回调, 一般在该方法中开始绘图
  * 实现的 SurfaceHolder.Callback 接口方法
  */
  @Override
  public void surfaceCreated(SurfaceHolder arg0) {
  System.out.println("SurfaceHolder.Callback.surfaceCreated : Surface 开始创建");
  }
  /**
  * 在 Surface 销毁之前回调, 在该方法中停止渲染线程, 释放相关资源
  * 实现的 SurfaceHolder.Callback 接口方法
  */
  @Override
  public void surfaceDestroyed(SurfaceHolder arg0) {
  System.out.println("SurfaceHolder.Callback.surfaceDestroyed : Surface 销毁");
  }
  @Override
  protected void onDestroy() {
  if(mediaPlayer != null)
    mediaPlayer.release();
  super.onDestroy();
  }
}



4. 运行示例


image.png





日志信息 :


octopus@octopus:~/develop/adt-bundle-linux/sdk/tools$ adb logcat -s System.out
--------- beginning of /dev/log/main
--------- beginning of /dev/log/system
I/System.out(21129): 设置数据源
I/System.out(21129): 视频播放长度 : 31200
I/System.out(21129): 准备完毕
I/System.out(21129): 缓冲了的百分比 : 0 %
I/System.out(21129): 缓冲了的百分比 : 37 %
I/System.out(21129): 缓冲了的百分比 : 37 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 缓冲了的百分比 : 100 %
I/System.out(21129): 播放完毕了
I/System.out(21129): 缓冲了的百分比 : 100 %
目录
相关文章
|
20天前
|
搜索推荐 前端开发 API
探索安卓开发中的自定义视图:打造个性化用户界面
在安卓应用开发的广阔天地中,自定义视图是一块神奇的画布,让开发者能够突破标准控件的限制,绘制出独一无二的用户界面。本文将带你走进自定义视图的世界,从基础概念到实战技巧,逐步揭示如何在安卓平台上创建和运用自定义视图来提升用户体验。无论你是初学者还是有一定经验的开发者,这篇文章都将为你打开新的视野,让你的应用在众多同质化产品中脱颖而出。
42 19
|
20天前
|
JSON Java API
探索安卓开发:打造你的首个天气应用
在这篇技术指南中,我们将一起潜入安卓开发的海洋,学习如何从零开始构建一个简单的天气应用。通过这个实践项目,你将掌握安卓开发的核心概念、界面设计、网络编程以及数据解析等技能。无论你是初学者还是有一定基础的开发者,这篇文章都将为你提供一个清晰的路线图和实用的代码示例,帮助你在安卓开发的道路上迈出坚实的一步。让我们一起开始这段旅程,打造属于你自己的第一个安卓应用吧!
46 14
|
23天前
|
Java Linux 数据库
探索安卓开发:打造你的第一款应用
在数字时代的浪潮中,每个人都有机会成为创意的实现者。本文将带你走进安卓开发的奇妙世界,通过浅显易懂的语言和实际代码示例,引导你从零开始构建自己的第一款安卓应用。无论你是编程新手还是希望拓展技术的开发者,这篇文章都将为你打开一扇门,让你的创意和技术一起飞扬。
|
21天前
|
XML 存储 Java
探索安卓开发之旅:从新手到专家
在数字时代,掌握安卓应用开发技能是进入IT行业的关键。本文将引导读者从零基础开始,逐步深入安卓开发的世界,通过实际案例和代码示例,展示如何构建自己的第一个安卓应用。我们将探讨基本概念、开发工具设置、用户界面设计、数据处理以及发布应用的全过程。无论你是编程新手还是有一定基础的开发者,这篇文章都将为你提供宝贵的知识和技能,帮助你在安卓开发的道路上迈出坚实的步伐。
31 5
|
20天前
|
开发框架 Android开发 iOS开发
安卓与iOS开发中的跨平台策略:一次编码,多平台部署
在移动应用开发的广阔天地中,安卓和iOS两大阵营各占一方。随着技术的发展,跨平台开发框架应运而生,它们承诺着“一次编码,到处运行”的便捷。本文将深入探讨跨平台开发的现状、挑战以及未来趋势,同时通过代码示例揭示跨平台工具的实际运用。
|
21天前
|
XML 搜索推荐 前端开发
安卓开发中的自定义视图:打造个性化UI组件
在安卓应用开发中,自定义视图是一种强大的工具,它允许开发者创造独一无二的用户界面元素,从而提升应用的外观和用户体验。本文将通过一个简单的自定义视图示例,引导你了解如何在安卓项目中实现自定义组件,并探讨其背后的技术原理。我们将从基础的View类讲起,逐步深入到绘图、事件处理以及性能优化等方面。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的见解和技巧。
|
21天前
|
搜索推荐 前端开发 测试技术
打造个性化安卓应用:从设计到开发的全面指南
在这个数字时代,拥有一个定制的移动应用不仅是一种趋势,更是个人或企业品牌的重要延伸。本文将引导你通过一系列简单易懂的步骤,从构思你的应用理念开始,直至实现一个功能齐全的安卓应用。无论你是编程新手还是希望拓展技能的开发者,这篇文章都将为你提供必要的工具和知识,帮助你将创意转化为现实。
|
24天前
|
Java 调度 Android开发
安卓与iOS开发中的线程管理差异解析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自拥有独特的魅力。如同东西方文化的差异,它们在处理多线程任务时也展现出不同的哲学。本文将带你穿梭于这两个平台之间,比较它们在线程管理上的核心理念、实现方式及性能考量,助你成为跨平台的编程高手。
|
21天前
|
Java Android开发 开发者
探索安卓开发:构建你的第一个“Hello World”应用
在安卓开发的浩瀚海洋中,每个新手都渴望扬帆起航。本文将作为你的指南针,引领你通过创建一个简单的“Hello World”应用,迈出安卓开发的第一步。我们将一起搭建开发环境、了解基本概念,并编写第一行代码。就像印度圣雄甘地所说:“你必须成为你希望在世界上看到的改变。”让我们一起开始这段旅程,成为我们想要见到的开发者吧!
28 0
|
24天前
|
存储 监控 Java
探索安卓开发:从基础到进阶的旅程
在这个数字时代,移动应用已成为我们日常生活的一部分。对于开发者来说,掌握安卓开发不仅是技能的提升,更是通往创新世界的钥匙。本文将带你了解安卓开发的核心概念,从搭建开发环境到实现复杂功能,逐步深入安卓开发的奥秘。无论你是初学者还是有经验的开发者,这篇文章都将为你提供新的见解和技巧,帮助你在安卓开发的道路上更进一步。
20 0