热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

Android使用Vitamio打造自己的万能播放器(10)——本地播放(缩略图、视频信息、视频扫描服务)

本文主要介绍Android使用Vitamio开发播放器,这里主要讲解本地播放(缩略图、视频信息、视频扫描服务)等功能,有需要的小伙伴可以参考下

前言

 Vitamio是我们团队的诚意之作,除了要将VPlayer打造成Android最好的播放器,也要将Vitamio打造成Android最好的播放器组件。新版发布内测SDK虽然还有一些问题,但已经具备高可扩展性、便捷性和许多内置实用的功能,后续文章将继续深挖和介绍Vitamio的使用细节和方法。本章将提供显示缩略图、视频大小等视频信息,并介绍使用Vitamio内置的ContentProvider以及文件扫描服务。

注意

 本章ContentProvider在2012-8-31发布的SDK中已经不能使用,后续文章会更新用法。

  系列

1、Android 使用Vitamio打造自己的万能播放器(1)——准备  

2、Android 使用Vitamio打造自己的万能播放器(2)—— 手势控制亮度、音量、缩放 

3、Android 使用Vitamio打造自己的万能播放器(3)——本地播放(主界面、视频列表) 

4、Android 使用Vitamio打造自己的万能播放器(4)——本地播放(快捷搜索、数据存储)

5、Android 使用Vitamio打造自己的万能播放器(5)——在线播放(播放优酷视频)

6、Android 使用Vitamio打造自己的万能播放器(6)——在线播放(播放列表)

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

8、Android 使用Vitamio打造自己的万能播放器(8)——细节优化

9、Android 使用Vitamio打造自己的万能播放器(9)——在线播放(在线电视)

正文

 一、目标

  1.1 获取视频缩略图等

   之前的文章都是自己扫描获取的视频文件信息,这里补充获取视频缩略图、视频宽带的代码,参加文章2.1部分。

  1.2 调用Vitamio的扫描服务

   新版SDK已经内置了视频扫描服务,提供了本地视频播放所需的全部数据。

 二、实现

  2.1 自己扫描获取视频的缩略图、视频宽高

 public static ArrayList batchBuildThumbnail(final Context ctx, final ArrayList files) {
 ArrayList result = new ArrayList();

 for (File f : files) {
 PFile pf = new PFile();
 try {
 if (f.exists() && f.canRead()) {
 //取出视频的一帧图像
 Bitmap bitmap = ThumbnailUtils.createVideoThumbnail(ctx, f.getAbsolutePath(), Video.Thumbnails.MINI_KIND);
 if (bitmap == null) {
 //缩略图创建失败
 bitmap = Bitmap.createBitmap(ThumbnailUtils.TARGET_SIZE_MINI_THUMBNAIL_WIDTH, ThumbnailUtils.TARGET_SIZE_MINI_THUMBNAIL_HEIGHT, Bitmap.Config.RGB_565);
 Log.e(TAG, "batchBuildThumbnail createBitmap faild : " + f.getAbsolutePath());
 }

 pf.width = bitmap.getWidth();
 pf.height = bitmap.getHeight();

 //缩略图
 bitmap = ThumbnailUtils.extractThumbnail(bitmap, ThumbnailUtils.dipToPX(ctx, ThumbnailUtils.TARGET_SIZE_MICRO_THUMBNAIL_WIDTH), ThumbnailUtils.dipToPX(ctx, ThumbnailUtils.TARGET_SIZE_MICRO_THUMBNAIL_HEIGHT), ThumbnailUtils.OPTIONS_RECYCLE_INPUT);
 if (bitmap != null) {
 File thum = new File(f.getParent(), f.getName() + ".jpg");
 pf.thumb = thum.getAbsolutePath();
 //thum.createNewFile();
 FileOutputStream iStream = new FileOutputStream(thum);
 bitmap.compress(Bitmap.CompressFormat.JPEG, 85, iStream);
 iStream.close();
 }

 if (bitmap != null)
 bitmap.recycle();
 }
 } catch (Exception e) {
 Log.e(TAG, e);
 continue;
 } finally {
 result.add(pf);
 }
 }

 return result;

 } 

代码说明:

    a). 缩略图的尺寸是可以调整的,大家可以看一下这个默认大小。

    b). 这里的代码是缩略图默认存在当前视频路径下面,并且文件名和视频名称一样,多加了一个jpg的后缀。

  2.2 调用Vitamio的视频扫描服务

   2.2.1  AndroidManifest.xml

 
 
 
 
 
 

 
 
 
 
 

 
 
 

 
 
 
 
 
 
 
 
 
 
 

代码说明:

     主要是MediaScannerService服务和MediaScannerReceiver监听器的注册。

   2.2.2  启动扫描服务(MainFragmentActivity

 if (!io.vov.vitamio.LibsChecker.checkVitamioLibs(this, getClass().getName(), R.string.init_decoders, R.raw.libarm))
 return;

 OPreference pref = new OPreference(this);
 //首次运行,扫描SD卡
 if (pref.getBoolean(PREF_KEY_FIRST, true)) {
 getApplicationContext().startService(new Intent(getApplicationContext(), MediaScannerService.class).putExtra(MediaScannerService.EXTRA_DIRECTORY, Environment.getExternalStorageDirectory().getAbsolutePath()));
 pref.putBooleanAndCommit(PREF_KEY_FIRST, false);
 }

代码说明:

     OPreference对象封装了SharedPreferences 存储数据。这里判断了是否首次启动应用,是的话启动视频扫描服务。

   2.2.3  获取数据(FragmentFile)

private static final String[] PROJECTION_MEDIA = new String[] { Video.Media._ID, Video.Media.TITLE, Video.Media.TITLE_KEY, Video.Media.SIZE, Video.Media.DURATION, Video.Media.DATA, Video.Media.WIDTH, Video.Media.HEIGHT };
 private static final String ORDER_MEDIA_TITLE = Video.Media.TITLE_KEY + " COLLATE NOCASE ASC";
 
 @Override
 public Loader onCreateLoader(int arg0, Bundle arg1) {
 Log.e(TAG, "onCreateLoader");
 return new CursorLoader(mParent, Video.Media.CONTENT_URI, PROJECTION_MEDIA, null, null, ORDER_MEDIA_TITLE);
 }

 @Override
 public void onLoadFinished(Loader arg0, Cursor newCursor) {
 Log.e(TAG, "onLoadFinished" + newCursor.getCount());
 mAdapter.swapCursor(newCursor);
 mListView.setSelectionFromTop(mVisiablePosition, mVisiableTop);
 }

 @Override
 public void onLoaderReset(Loader arg0) {
 Log.e(TAG, "onLoaderReset");
 mAdapter.swapCursor(null);
 }

代码说明:

   a). FragmentFile继承了LoaderCallbacks,相关的资料大家可以搜一下。

   b). 其中Video.Media.DATA 存的是视频的路径。

   c). 取缩略图的方法:

     Video.Thumbnails.getThumbnail(ctx.getApplicationContext(), ctx.getContentResolver(), _id, Video.Thumbnails.MICRO_KIND, null)

    这个"_id"就是Video.Media._ID,缩略图存在SD里,大家可以自己加一些缓存处理。

 三、注意事项

   3.1 启动视频服务扫描前,需要先确保解码库已经被解压,即调用代码:LibsChecker.checkVitamioLibs

   3.2 确保MediaScannerService与当前应用程序在一个进程中,即不要指定其android:process,否则无法使用视频扫描服务。

   3.3 目前获取扫描完成广播接收还有问题,即无法获知是否已经扫描完成。目前的办法是每次数据变更后5秒再去判断是否有该Service,没有了就表示已经扫描完成。参见方法MediaScannerReceiver.isServiceRunning。

  解码库

   https://bitbucket.org/ABitNo/ffmpegandroid

   http://vplayer.net      的首页底部已经给出链接,注意代码并不会实时同步,有需要的朋友可以参考一下。

 四、OPlayer

  4.1 下载

   请移步#Taocode(SVN):

   项目地址:http://code.taobao.org/p/oplayer

   SVN地址:http://code.taobao.org/svn/oplayer/

  4.2 更新

   a). 移动显示SD剩余容量至底部,并在其左边显示正在加载的图标。

   b). 保留原FragmentFile为FragmentFildOld,新版已经切换至使用Vitamio内置的视频扫描服务。

   c). 更换Adapter为CursorAdapter,同时更新A-Z的功能,参加代码更新。

   d). 新增了Preference,用于存储一般设置数据,例如是否首次启动。

 五、补充说明

  已经在SDK发布版本里面强调了许多,这里再强调一下:

  5.1 Vitamio相关项目不能同时存在同一个设备上,也不能与VPlayer共存,这是一个已知的BUG,下一版本会解决。

  5.2 Vitamio的示例工程需要在API Level14以上编译,但是最低版本可以支持7以上(android:minSdkVersion="7")。可以把values-v11和values-v14两个文件夹删除,Vitamio在目前项目使用中是兼容Android2.1以上版本,更低的没有试过,大家可以试一下。

  5.3 网络视频播放不了、很慢可能有以下几个原因:

   a). 你的网速不行

   b). 视频服务器不行,或者其带宽不够

   c). 可能是播放组件的问题,我们还在持续的改进,加大再网络播放这部分的优化,感谢理解和支持。

    遇到这种情况的时候,如果方便请把视频测试地址发给我们,并附上简要说明。(vplayer@yixia.com)

  5.4 一直停留在初始化解码包界面的问题

   一般出现这种情况,只有一个原因:你的设备上还有其他Vitamio的相关项目,参照5.1。

结束

 这篇文章耗费了我至少10个小时以上,昨晚熬到2点,为的是解决使用者心中的疑虑,最终更好的服务于用户。在文档不够全的情况下通过这种方式来与大家分享使用方法和技巧以及注意点,还请大家谅解。


推荐阅读
  • 如何高效学习鸿蒙操作系统:开发者指南
    本文探讨了开发者如何更有效地学习鸿蒙操作系统,提供了来自行业专家的建议,包括系统化学习方法、职业规划建议以及具体的开发技巧。 ... [详细]
  • Redis: 高效的键值存储系统
    Redis是一款遵循BSD许可的开源高性能键值存储系统,它不仅支持多种数据类型的存储,还提供了数据持久化和复制等功能,显著区别于其他键值缓存解决方案。 ... [详细]
  • 本文介绍了一个基本的同步Socket程序,演示了如何实现客户端与服务器之间的简单消息传递。此外,文章还概述了Socket的基本工作流程,并计划在未来探讨同步与异步Socket的区别。 ... [详细]
  • 使用IntelliJ IDEA高效开发与运行Shell脚本
    本文介绍了如何利用IntelliJ IDEA中的BashSupport插件来增强Shell脚本的开发体验,包括插件的安装、配置以及脚本的运行方法。 ... [详细]
  • 本文将详细介绍如何配置并整合MVP架构、Retrofit网络请求库、Dagger2依赖注入框架以及RxAndroid响应式编程库,构建高效、模块化的Android应用。 ... [详细]
  • Hadoop集群搭建:实现SSH无密码登录
    本文介绍了如何在CentOS 7 64位操作系统环境下配置Hadoop集群中的SSH无密码登录,包括环境准备、用户创建、密钥生成及配置等步骤。 ... [详细]
  • 浅谈Android五大布局——LinearLayout、FrameLayout和AbsoulteLa
    为什么80%的码农都做不了架构师?Android的界面是有布局和组件协同完成的,布局好比是建筑里的框架,而组件则相当于建筑里的砖瓦。 ... [详细]
  • LoadRunner中的IP欺骗配置与实践
    为了确保服务器能够有效地区分不同的用户请求,避免多人使用同一IP地址造成的访问限制,可以通过配置IP欺骗来解决这一问题。本文将详细介绍IP欺骗的工作原理及其在LoadRunner中的具体配置步骤。 ... [详细]
  • 本文详细介绍了PHP中的几种超全局变量,包括$GLOBAL、$_SERVER、$_POST、$_GET等,并探讨了AJAX的工作原理及其优缺点。通过具体示例,帮助读者更好地理解和应用这些技术。 ... [详细]
  • 本文概述了在GNU/Linux系统中,动态库在链接和运行阶段的搜索路径及其指定方法,包括通过编译时参数、环境变量及系统配置文件等方式来控制动态库的查找路径。 ... [详细]
  • 使用 ModelAttribute 实现页面数据自动填充
    本文介绍了如何利用 Spring MVC 中的 ModelAttribute 注解,在页面跳转后自动填充表单数据。主要探讨了两种实现方法及其背后的原理。 ... [详细]
  • 2023年1月28日网络安全热点
    涵盖最新的网络安全动态,包括OpenSSH和WordPress的安全更新、VirtualBox提权漏洞、以及谷歌推出的新证书验证机制等内容。 ... [详细]
  • Docker基础入门与环境配置指南
    本文介绍了Docker——一款用Go语言编写的开源应用程序容器引擎。通过Docker,用户能够将应用及其依赖打包进容器内,实现高效、轻量级的虚拟化。容器之间采用沙箱机制,确保彼此隔离且资源消耗低。 ... [详细]
  • 本文详细介绍了如何在PHP中使用Memcached进行数据缓存,包括服务器连接、数据操作、高级功能等。 ... [详细]
  • 解决Expo XDE 2.22.1版本启动错误
    根据问题描述,用户在将Expo升级至2.22.1版本后,在尝试打开项目时遇到了错误。本文提供了详细的错误分析及解决方案。 ... [详细]
author-avatar
85中互动_715
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有