使用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
目录
相关文章
|
1天前
|
Java 程序员 开发工具
Android|修复阿里云播放器下载不回调的问题
虽然 GC 带来了很多便利,但在实际编码时,我们也需要注意对象的生命周期管理,该存活的存活,该释放的释放,避免因为 GC 导致的问题。
12 2
|
3月前
|
Android开发
Android 利用MediaPlayer实现音乐播放
本文提供了一个简单的Android MediaPlayer音乐播放示例,包括创建PlayerActivity、配置AndroidManifest.xml和activity_player.xml布局,以及实现播放和暂停功能的代码。
24 0
Android 利用MediaPlayer实现音乐播放
|
3月前
|
编解码 网络协议 开发工具
Android平台如何实现多路低延迟RTSP|RTMP播放?
本文档详细介绍了大牛直播SDK在Android平台上实现RTSP与RTMP流媒体播放及录像功能的技术细节。早在2015年,SDK的第一版就已经支持了多实例播放,并且通过简单的实例封装就能轻松实现。文档中提供了代码示例,展示了如何开启播放、停止播放以及开始和停止录像等功能。此外,SDK还提供了丰富的配置选项,例如设置录像目录、文件大小限制、转码选项等。总结部分列出了该SDK的关键特性,包括但不限于高稳定性和低延迟的播放能力、多实例支持、事件回调、硬解码支持、网络状态监控以及复杂的网络环境处理等。这些功能使得SDK能够应对各种应用场景,特别是在对延迟和稳定性有极高要求的情况下表现优异。
|
3月前
|
编解码 网络协议 vr&ar
Android平台下VR头显如何低延迟播放4K以上超高分辨率RTSP|RTMP流
这段内容讲述了VR头显中实现高分辨率视频播放的技术背景与实现方法,并强调了其重要性。高分辨率对于提升VR体验至关重要,它能提供更清晰的画面、增强沉浸感、补偿透镜放大效应,并维持宽广视场角下的图像质量。文中提到的大牛直播SDK具备极低的延迟(200-400ms),支持多种协议与格式,并具有丰富的功能特性,如多实例播放、事件回调、视频及音频格式支持等。此外,提供了基于Unity的播放器示例代码,展示了如何配置播放参数并开始播放。最后,作者指出此类技术在远程控制、虚拟仿真等应用场景中的重要意义。
|
3月前
|
编解码 开发工具 Android开发
Android平台RTSP|RTMP播放器如何实现TextureView渲染
本文介绍了在Android平台上使用TextureView进行RTSP和RTMP视频流渲染的技术背景和实现方法。TextureView相较于SurfaceView具备更高性能、更强功能性和更灵活的绘制方式等优势,但也有必须在硬件加速环境下运行和较高内存占用等局限。文中详细展示了如何在SmartPlayerV2工程中创建和配置TextureView,并通过代码示例解释了如何根据视频分辨率信息调整显示比例,以及处理TextureView的各种生命周期回调。此外,还列举了该播放器SDK支持的多项高级功能,如多实例播放、多种编码格式支持、硬解码能力等,旨在帮助开发者更好地理解和实现高性能的直播播放器。
|
3月前
|
编解码 监控 API
惊艳登场!揭秘如何在Android平台上轻松玩转GB28181标准,实现多视频通道接入的超实用指南!
【8月更文挑战第14天】GB28181是公共安全视频监控联网的技术标准。本文介绍如何在Android平台上实现该标准下的多视频通道接入。首先准备开发环境,接着引入GB28181 SDK依赖并初始化SDK。实现设备注册与登录后,通过指定不同通道号请求多路视频流。最后,处理接收到的数据并显示给用户。此过程涉及视频解码,需确保应用稳定及良好的用户体验。
27 0
|
3月前
|
编解码 网络协议 开发工具
Android平台RTSP|RTMP直播播放器技术接入说明
大牛直播SDK自2015年发布RTSP、RTMP直播播放模块,迭代从未停止,SmartPlayer功能强大、性能强劲、高稳定、超低延迟、超低资源占用。无需赘述,全自研内核,行业内一致认可的跨平台RTSP、RTMP直播播放器。本文以Android平台为例,介绍下如何集成RTSP、RTMP播放模块。
143 0
|
18天前
|
缓存 搜索推荐 Android开发
安卓开发中的自定义控件实践
【10月更文挑战第4天】在安卓开发的海洋中,自定义控件是那片璀璨的星辰。它不仅让应用界面设计变得丰富多彩,还提升了用户体验。本文将带你探索自定义控件的核心概念、实现过程以及优化技巧,让你的应用在众多竞争者中脱颖而出。
|
19天前
|
Java Android开发 Swift
安卓与iOS开发对比:平台选择对项目成功的影响
【10月更文挑战第4天】在移动应用开发的世界中,选择合适的平台是至关重要的。本文将深入探讨安卓和iOS两大主流平台的开发环境、用户基础、市场份额和开发成本等方面的差异,并分析这些差异如何影响项目的最终成果。通过比较这两个平台的优势与挑战,开发者可以更好地决定哪个平台更适合他们的项目需求。
70 1
|
19天前
|
Web App开发 安全 程序员
FFmpeg开发笔记(五十五)寒冬里的安卓程序员可进阶修炼的几种姿势
多年的互联网寒冬在今年尤为凛冽,坚守安卓开发愈发不易。面对是否转行或学习新技术的迷茫,安卓程序员可从三个方向进阶:1)钻研谷歌新技术,如Kotlin、Flutter、Jetpack等;2)拓展新功能应用,掌握Socket、OpenGL、WebRTC等专业领域技能;3)结合其他行业,如汽车、游戏、安全等,拓宽职业道路。这三个方向各有学习难度和保饭碗指数,助你在安卓开发领域持续成长。
48 1
FFmpeg开发笔记(五十五)寒冬里的安卓程序员可进阶修炼的几种姿势