使用Vitamio打造自己的Android万能播放器(7)——在线播放(下载视频)

简介:

前言

  本章将实现非常实用的功能——下载在线视频。涉及到多线程、线程更新UI等技术,还需思考产品的设计,如何将新加的功能更好的融入到现有的产品中,并不是简单的加一个界面就行了,欢迎大家交流产品设计和技术细节实现!
 
声明
  欢迎转载,但请保留文章原始出处:) 
    博客园:http://www.cnblogs.com
    农民伯伯: http://over140.cnblogs.com 
 
系列
  4、 使用Vitamio打造自己的Android万能播放器(4)——本地播放(快捷搜索、数据存储)
  5、 使用Vitamio打造自己的Android万能播放器(5)——在线播放(播放优酷视频)

 

正文
  一、目标

    本章实现视频下载的功能

       

    使用说明:进入在线视频,点击播放时将弹出选择框询问播放还是下载,点击下载后进度条将在本地视频顶部显示。如果想边看便下载,请直接点击本地播放列表中正在下载的视频。

 

  二、实现(部分主要实现代码)
    FileDownloadHelper
复制代码
public  class FileDownloadHelper {
     private  static  final String TAG = "FileDownloadHelper";
     /**  线程池  */
     private ThreadPool mPool =  new ThreadPool();
     /**  开始下载  */
     public  static  final  int MESSAGE_START = 0;
     /**  更新进度  */
     public  static  final  int MESSAGE_PROGRESS = 1;
     /**  下载结束  */
     public  static  final  int MESSAGE_STOP = 2;
     /**  下载出错  */
     public  static  final  int MESSAGE_ERROR = 3;
     /**  中途终止  */
     private  volatile  boolean mIsStop =  false;
     private Handler mHandler;
     public  volatile HashMap<String, String> mDownloadUrls =  new HashMap<String, String>();

     public FileDownloadHelper(Handler handler) {
         if (handler ==  null)
             throw  new IllegalArgumentException("handler不能为空!");

         this.mHandler = handler;
    }

     public  void stopALl() {
        mIsStop =  true;
        mPool.stop();
    }

     public  void newDownloadFile( final String url) {
        newDownloadFile(url, Environment.getExternalStorageDirectory() + "/" + FileUtils.getUrlFileName(url));
    }

     /**
     * 下载一个新的文件
     * 
     * 
@param  url
     * 
@param  savePath
     
*/
     public  void newDownloadFile( final String url,  final String savePath) {
         if (mDownloadUrls.containsKey(url))
             return;
         else
            mDownloadUrls.put(url, savePath);
        mPool.start( new Runnable() {

            @Override
             public  void run() {
                mHandler.sendMessage(mHandler.obtainMessage(MESSAGE_START, url));
                HttpClient client =  new DefaultHttpClient();
                HttpGet get =  new HttpGet(url);
                InputStream inputStream =  null;
                FileOutputStream outputStream =  null;
                 try {
                    HttpResponse response = client.execute(get);
                    HttpEntity entity = response.getEntity();
                     final  int size = ( int) entity.getContentLength();
                    inputStream = entity.getContent();
                     if (size > 0 && inputStream !=  null) {
                        outputStream =  new FileOutputStream(savePath);
                         int ch = -1;
                         byte[] buf =  new  byte[1024];
                         // 每秒更新一次进度
                         new Timer().schedule( new TimerTask() {

                            @Override
                             public  void run() {
                                 try {
                                    FileInputStream fis =  new FileInputStream( new File(savePath));
                                     int downloadedSize = fis.available();
                                     if (downloadedSize >= size)
                                        cancel();
                                    mHandler.sendMessage(mHandler.obtainMessage(MESSAGE_PROGRESS, downloadedSize, size, url));
                                }  catch (Exception e) {

                                }
                            }
                        }, 50, 1000);

                         while ((ch = inputStream.read(buf)) != -1 && !mIsStop) {
                            outputStream.write(buf, 0, ch);
                        }
                        outputStream.flush();
                    }
                }  catch (Exception e) {
                    Log.e(TAG, e.getMessage(), e);
                    mHandler.sendMessage(mHandler.obtainMessage(MESSAGE_ERROR, url + ":" + e.getMessage()));
                }  finally {
                     try {
                         if (outputStream !=  null)
                            outputStream.close();
                    }  catch (IOException ex) {
                    }
                     try {
                         if (inputStream !=  null)
                            inputStream.close();
                    }  catch (IOException ex) {
                    }
                }
                mDownloadUrls.remove(url);
                mHandler.sendMessage(mHandler.obtainMessage(MESSAGE_STOP, url));
            }
        });
    }
复制代码

    代码说明:
      a. ThreadPool是线程池,请参照项目代码。
      b. 这里使用了Time定时来刷进度,而没有直接在write数据时更新进度,这样的原因时每秒write较高,更新UI过于频繁,可能导致超时等问题。

    Handle

复制代码
     public Handler mDownloadHandler =  new Handler() {
        @Override
         public  void handleMessage(Message msg) {
            PFile p;
            String url = msg.obj.toString();
             switch (msg.what) {
             case FileDownloadHelper.MESSAGE_START: // 开始下载
                p =  new PFile();
                p.path = mParent.mFileDownload.mDownloadUrls.get(url);
                p.title =  new File(p.path).getName();
                p.status = 0;
                p.file_size = 0;
                 if (mDownloadAdapter ==  null) {
                    mDownloadAdapter =  new FileAdapter(getActivity(),  new ArrayList<PFile>());
                    mDownloadAdapter.add(p, url);
                    mTempListView.setAdapter(mDownloadAdapter);
                    mTempListView.setVisibility(View.VISIBLE);
                }  else {
                    mDownloadAdapter.add(p, url);
                    mDownloadAdapter.notifyDataSetChanged();
                }
                 break;
             case FileDownloadHelper.MESSAGE_PROGRESS: // 正在下载
                p = mDownloadAdapter.getItem(url);
                p.temp_file_size = msg.arg1;
                p.file_size = msg.arg2;
                 int status = ( int) ((msg.arg1 * 1.0 / msg.arg2) * 10);
                 if (status > 10)
                    status = 10;
                p.status = status;
                mDownloadAdapter.notifyDataSetChanged();
                 break;
             case FileDownloadHelper.MESSAGE_STOP: // 下载结束
                p = mDownloadAdapter.getItem(url);
                FileBusiness.insertFile(getActivity(), p);
                 break;
             case FileDownloadHelper.MESSAGE_ERROR:
                Toast.makeText(getActivity(), url, Toast.LENGTH_LONG).show();
                 break;
            }
             super.handleMessage(msg);
        }
复制代码

    }; 

    代码说明:
      a. mTempListView是新增的,默认是隐藏,请参见项目代码layout部分。
      b. 下载流程:开始(显示mTempListView) -> 正在下载(更新进度图片和大小)  -> 完成(入裤)

    Dialog

复制代码
                 if (FileUtils.isVideoOrAudio(url)) {
                    Dialog dialog =  new AlertDialog.Builder(getActivity()).setIcon(android.R.drawable.btn_star).setTitle("播放/下载").setMessage(url).setPositiveButton("播放",  new OnClickListener() {
                        @Override
                         public  void onClick(DialogInterface dialog,  int which) {
                            Intent intent =  new Intent(getActivity(), VideoPlayerActivity. class);
                            intent.putExtra("path", url);
                            startActivity(intent);
                        }
                    }).setNeutralButton("下载",  new OnClickListener() {
                        @Override
                         public  void onClick(DialogInterface dialog,  int which) {
                            MainFragmentActivity activity = (MainFragmentActivity) getActivity();
                            activity.mFileDownload.newDownloadFile(url);
                            Toast.makeText(getActivity(), "正在下载 .." + FileUtils.getUrlFileName(url) + " ,可从本地视频查看进度!", Toast.LENGTH_LONG).show();
                        }
                    }).setNegativeButton("取消",  null).create();
                    dialog.show();
                     return  true;
复制代码

                } 

 

  三、下载
     至本章节往后,代码均不再提供下载,请移步 Google Code:
      http://code.google.com/p/android-oplayer

 

  四、Vitamio公告

      正式建立Vitamio开发者联盟QQ群!群号为:246969281

      注意: 目前仅接受已经开发基于Vitamio产品的开发者申请加入 ,申请理由请填写产品的名词和链接,获取最新进展以及与Vitamio作者直接交流机会!

 

结束
  有BUG不可怕,改了就行,大胆设计、放手写代码,谨慎处理已知细节,这样的软件才会越来越好。写了一上午代码,难免有出错的地方,欢迎反馈~

转载:http://www.cnblogs.com/over140/archive/2012/06/29/2561244.html
目录
相关文章
|
5月前
|
Ubuntu 开发工具 Android开发
Repo下载AOSP源码:基于ubuntu22.04 环境配置,android-12.0.0_r32
本文介绍了在基于Ubuntu 22.04的环境下配置Python 3.9、安装repo工具、下载和同步AOSP源码包以及处理repo同步错误的详细步骤。
294 0
Repo下载AOSP源码:基于ubuntu22.04 环境配置,android-12.0.0_r32
|
3月前
|
Java 程序员 开发工具
Android|修复阿里云播放器下载不回调的问题
虽然 GC 带来了很多便利,但在实际编码时,我们也需要注意对象的生命周期管理,该存活的存活,该释放的释放,避免因为 GC 导致的问题。
47 2
|
5月前
|
API 开发工具 Android开发
Android源码下载
Android源码下载
588 0
|
5月前
|
开发工具 uml git
AOSP源码下载方法,解决repo sync错误:android-13.0.0_r82
本文分享了下载AOSP源码的方法,包括如何使用repo工具和处理常见的repo sync错误,以及配置Python环境以确保顺利同步特定版本的AOSP代码。
653 0
AOSP源码下载方法,解决repo sync错误:android-13.0.0_r82
|
5月前
|
编解码 开发工具 Android开发
Android平台RTSP|RTMP播放器如何实现TextureView渲染
本文介绍了在Android平台上使用TextureView进行RTSP和RTMP视频流渲染的技术背景和实现方法。TextureView相较于SurfaceView具备更高性能、更强功能性和更灵活的绘制方式等优势,但也有必须在硬件加速环境下运行和较高内存占用等局限。文中详细展示了如何在SmartPlayerV2工程中创建和配置TextureView,并通过代码示例解释了如何根据视频分辨率信息调整显示比例,以及处理TextureView的各种生命周期回调。此外,还列举了该播放器SDK支持的多项高级功能,如多实例播放、多种编码格式支持、硬解码能力等,旨在帮助开发者更好地理解和实现高性能的直播播放器。
|
5月前
|
算法 数据处理 开发工具
Android平台RTSP|RTMP播放器如何回调YUV或RGB数据
在开发Android平台上的RTSP或RTMP播放器时,开发者不仅追求低延迟播放,还希望获取解码后的视频数据(如YUV或RGB格式),以便进行视觉算法分析。使用大牛直播SDK中的SmartPlayer,可在确保播放流畅的同时,通过设置外部渲染器(`SmartPlayerSetExternalRender`)来高效地回调原始视频数据。例如,对于RGBA数据,需实现`NTExternalRender`接口,并重写相关方法以处理数据和尺寸变化。同样地,对于I420(YUV)数据,也需要相应地实现接口以满足需求。这种方式使得开发者能在不影响常规播放功能的情况下,进行定制化的视频处理任务。
|
5月前
|
存储 监控 数据库
Android经典实战之OkDownload的文件分段下载及合成原理
本文介绍了 OkDownload,一个高效的 Android 下载引擎,支持多线程下载、断点续传等功能。文章详细描述了文件分段下载及合成原理,包括任务创建、断点续传、并行下载等步骤,并展示了如何通过多种机制保证下载的稳定性和完整性。
153 0
|
5月前
|
Ubuntu 开发工具 Android开发
Repo下载、编译AOSP源码:基于Ubuntu 21.04,android-12.1.0_r27
文章记录了作者在Ubuntu 21.04服务器上配置环境、下载并编译基于Android 12.1.0_r27版本的AOSP源码的过程,包括解决编译过程中遇到的问题和错误处理方法。
260 0
|
5月前
|
编解码 监控 API
惊艳登场!揭秘如何在Android平台上轻松玩转GB28181标准,实现多视频通道接入的超实用指南!
【8月更文挑战第14天】GB28181是公共安全视频监控联网的技术标准。本文介绍如何在Android平台上实现该标准下的多视频通道接入。首先准备开发环境,接着引入GB28181 SDK依赖并初始化SDK。实现设备注册与登录后,通过指定不同通道号请求多路视频流。最后,处理接收到的数据并显示给用户。此过程涉及视频解码,需确保应用稳定及良好的用户体验。
111 0