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

简介:
一、目标

    本章实现视频下载的功能

       

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

    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打造自己的Android万能播放器(7)——在线播放(下载视频),如需转载请自行联系原博主。


目录
相关文章
|
4月前
|
XML 存储 数据库
如何使用Android Studio创建一个基本的音乐播放器应用
如何使用Android Studio创建一个基本的音乐播放器应用
156 0
|
2月前
|
编解码 开发工具 Android开发
Android平台RTSP|RTMP播放器如何实现TextureView渲染
本文介绍了在Android平台上使用TextureView进行RTSP和RTMP视频流渲染的技术背景和实现方法。TextureView相较于SurfaceView具备更高性能、更强功能性和更灵活的绘制方式等优势,但也有必须在硬件加速环境下运行和较高内存占用等局限。文中详细展示了如何在SmartPlayerV2工程中创建和配置TextureView,并通过代码示例解释了如何根据视频分辨率信息调整显示比例,以及处理TextureView的各种生命周期回调。此外,还列举了该播放器SDK支持的多项高级功能,如多实例播放、多种编码格式支持、硬解码能力等,旨在帮助开发者更好地理解和实现高性能的直播播放器。
|
2月前
|
算法 数据处理 开发工具
Android平台RTSP|RTMP播放器如何回调YUV或RGB数据
在开发Android平台上的RTSP或RTMP播放器时,开发者不仅追求低延迟播放,还希望获取解码后的视频数据(如YUV或RGB格式),以便进行视觉算法分析。使用大牛直播SDK中的SmartPlayer,可在确保播放流畅的同时,通过设置外部渲染器(`SmartPlayerSetExternalRender`)来高效地回调原始视频数据。例如,对于RGBA数据,需实现`NTExternalRender`接口,并重写相关方法以处理数据和尺寸变化。同样地,对于I420(YUV)数据,也需要相应地实现接口以满足需求。这种方式使得开发者能在不影响常规播放功能的情况下,进行定制化的视频处理任务。
|
2月前
|
编解码 网络协议 开发工具
Android平台RTSP|RTMP直播播放器技术接入说明
大牛直播SDK自2015年发布RTSP、RTMP直播播放模块,迭代从未停止,SmartPlayer功能强大、性能强劲、高稳定、超低延迟、超低资源占用。无需赘述,全自研内核,行业内一致认可的跨平台RTSP、RTMP直播播放器。本文以Android平台为例,介绍下如何集成RTSP、RTMP播放模块。
120 0
|
4月前
|
API Android开发 UED
56. 【Android教程】媒体播放器:MediaPlayer
56. 【Android教程】媒体播放器:MediaPlayer
85 0
|
5月前
|
存储 数据库 Android开发
|
5月前
|
XML 存储 Java
Android 开发音频录播中媒体录制器MediaRecorder和媒体播放器MediaPlayer的讲解及实战(超详细 附源码)
Android 开发音频录播中媒体录制器MediaRecorder和媒体播放器MediaPlayer的讲解及实战(超详细 附源码)
104 0
|
10天前
|
XML 存储 Java
探索安卓开发之旅:从基础到进阶
【9月更文挑战第37天】安卓开发,一个充满无限可能的领域。它不仅关乎技术的深度与广度,更关乎开发者的成长与突破。本文将带你走进安卓开发的世界,从基础知识的学习到进阶技巧的掌握,一起感受编程的魅力与乐趣。
|
4天前
|
缓存 搜索推荐 Android开发
安卓开发中的自定义控件实践
【10月更文挑战第4天】在安卓开发的海洋中,自定义控件是那片璀璨的星辰。它不仅让应用界面设计变得丰富多彩,还提升了用户体验。本文将带你探索自定义控件的核心概念、实现过程以及优化技巧,让你的应用在众多竞争者中脱颖而出。