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

前言

  本章继续完善播放相关播放器的核心功能,为后续扩展打好基础。

系列

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

正文

一、实现目标

1.1 亮度控制

模仿VPlayer界面:

1.2 声音控制

模仿VPlayer界面:

1.3 画面缩放

根据下面API提供画面的拉伸、剪切、100%、全屏

    二、Vitamio API 介绍

VideoView

2.1 public void start()

开始播放

2.2 public void pause()

暂停播放

2.3 public long getDuration()

获取视频的时长

2.4 public long getCurrentPosition()

获取已经播放的时长

2.5 public void seekTo(long msec)

设置播放器从指定的位置开始播放

2.6 public boolean isPlaying()

是否正在播放

2.7 public int getVideoWidth()

获取视频宽

2.8 public int getVideoHeight()

获取视频高

2.9 public void setBufferSize(int bufSize)

设置缓存大小,默认1024KB

2.10 public void setVideoQuality(int quality)

设置视频质量,低、中、高(MediaPlayer.VIDEOQUALITY_LOW、MediaPlayer.VIDEOQUALITY_MEDIUM 、MediaPlayer.VIDEOQUALITY_HIGH ),

默认低(最流畅)。

2.11 public void setSubShown(boolean shown)

设置是否显示字幕

2.12 public void setAudioTrack(int audioIndex)

设置音轨,必须是getAudioTrackMap(String) 的返回值。

2.13 public void setVolume(float leftVolume, float rightVolume)

设置立体音左右音量。

2.14 public void setSubPath(String subPath)

设置外挂字幕路径

2.15 public int getBufferPercentage()

获取缓冲百分比

2.16 public void stopPlayback()

停止播放

2.17 public void setVideoPath(String path)

设置视频播放路径

2.18 public void setVideoURI(Uri uri)

设置视频播放路径

2.19 public void setVideoLayout(int layout, float aspectRatio)

设置视频缩放(拉伸、剪切、100%、全屏)

     三、 实现代码

3.1 xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
 android:orientation="vertical" android:layout_width="match_parent"
 android:layout_height="match_parent">
 <io.vov.vitamio.widget.VideoView
 android:id="@+id/surface_view" android:layout_width="match_parent"
 android:layout_height="match_parent" android:layout_centerHorizontal="true"
 android:layout_centerVertical="true" />
 <FrameLayout android:id="@+id/operation_volume_brightness"
 android:visibility="invisible" android:layout_centerInParent="true"
 android:layout_width="wrap_content" android:layout_height="wrap_content"
 android:background="#00000000" android:orientation="horizontal"
 android:padding="0dip">
 <ImageView android:id="@+id/operation_bg"
 android:layout_gravity="center" android:src="@drawable/video_volumn_bg"
 android:layout_width="wrap_content" android:layout_height="wrap_content" />
 <FrameLayout android:layout_gravity="bottom|center_horizontal"
 android:layout_width="wrap_content" android:layout_height="wrap_content"
 android:paddingBottom="25dip">
 <ImageView android:id="@+id/operation_full"
 android:layout_gravity="left" android:src="@drawable/video_num_bg"
 android:layout_width="94dip" android:layout_height="wrap_content" />
 <ImageView android:id="@+id/operation_percent"
 android:layout_gravity="left" android:src="@drawable/video_num_front"
 android:layout_width="0dip" android:layout_height="wrap_content"
 android:scaleType="matrix" />
 </FrameLayout>
 </FrameLayout>
</RelativeLayout>

3.2 Activity

 /**
 *
 * Android万能播放器
 *
 *
 */
 public class VideoViewDemo extends Activity {

 private String path = Environment.getExternalStorageDirectory()
 + "/Moon.mp4";
 private VideoView mVideoView;
 private View mVolumeBrightnessLayout;
 private ImageView mOperationBg;
 private ImageView mOperationPercent;
 private AudioManager mAudioManager;
 /** 最大声音 */
 private int mMaxVolume;
 /** 当前声音 */
 private int mVolume = -1;
 /** 当前亮度 */
 private float mBrightness = -1f;
 /** 当前缩放模式 */
 private int mLayout = VideoView.VIDEO_LAYOUT_ZOOM;
 private GestureDetector mGestureDetector;
 private MediaController mMediaController;

 @Override
 public void onCreate(Bundle icicle) {
 super.onCreate(icicle);
 setContentView(R.layout.videoview);
 mVideoView = (VideoView) findViewById(R.id.surface_view);
 mVolumeBrightnessLayout = findViewById(R.id.operation_volume_brightness);
 mOperationBg = (ImageView) findViewById(R.id.operation_bg);
 mOperationPercent = (ImageView) findViewById(R.id.operation_percent);

 mAudioManager = (AudioManager) getSystemService(Context.AUDIO_SERVICE);
 mMaxVolume = mAudioManager
 .getStreamMaxVolume(AudioManager.STREAM_MUSIC);
 mVideoView.setVideoPath(path);
 mMediaController = new MediaController(this);
 mVideoView.setMediaController(mMediaController);
 mVideoView.requestFocus();

 mGestureDetector = new GestureDetector(this, new MyGestureListener());
 }

 @Override
 public boolean onTouchEvent(MotionEvent event) {
 if (mGestureDetector.onTouchEvent(event))
 return true;

 // 处理手势结束
 switch (event.getAction() & MotionEvent.ACTION_MASK) {
 case MotionEvent.ACTION_UP:
 endGesture();
 break;
 }

 return super.onTouchEvent(event);
 }

 /** 手势结束 */
 private void endGesture() {
 mVolume = -1;
 mBrightness = -1f;

 // 隐藏
 mDismissHandler.removeMessages(0);
 mDismissHandler.sendEmptyMessageDelayed(0, 500);
 }

 private class MyGestureListener extends SimpleOnGestureListener {

 /** 双击 */
 @Override
 public boolean onDoubleTap(MotionEvent e) {
 if (mLayout == VideoView.VIDEO_LAYOUT_ZOOM)
 mLayout = VideoView.VIDEO_LAYOUT_ORIGIN;
 else
 mLayout++;
 if (mVideoView != null)
 mVideoView.setVideoLayout(mLayout, 0);
 return true;
 }

 /** 滑动 */
 @Override
 public boolean onScroll(MotionEvent e1, MotionEvent e2,
 float distanceX, float distanceY) {
 float mOldX = e1.getX(), mOldY = e1.getY();
 int y = (int) e2.getRawY();
 Display disp = getWindowManager().getDefaultDisplay();
 int windowWidth = disp.getWidth();
 int windowHeight = disp.getHeight();

 if (mOldX > windowWidth * 4.0 / 5)// 右边滑动
 onVolumeSlide((mOldY - y) / windowHeight);
 else if (mOldX < windowWidth / 5.0)// 左边滑动
 onBrightnessSlide((mOldY - y) / windowHeight);

 return super.onScroll(e1, e2, distanceX, distanceY);
 }
 }

 /** 定时隐藏 */
 private Handler mDismissHandler = new Handler() {
 @Override
 public void handleMessage(Message msg) {
 mVolumeBrightnessLayout.setVisibility(View.GONE);
 }
 };

 /**
 * 滑动改变声音大小
 *
 * @param percent
 */
 private void onVolumeSlide(float percent) {
 if (mVolume == -1) {
 mVolume = mAudioManager.getStreamVolume(AudioManager.STREAM_MUSIC);
 if (mVolume < 0)
 mVolume = 0;

 // 显示
 mOperationBg.setImageResource(R.drawable.video_volumn_bg);
 mVolumeBrightnessLayout.setVisibility(View.VISIBLE);
 }

 int index = (int) (percent * mMaxVolume) + mVolume;
 if (index > mMaxVolume)
 index = mMaxVolume;
 else if (index < 0)
 index = 0;

 // 变更声音
 mAudioManager.setStreamVolume(AudioManager.STREAM_MUSIC, index, 0);

 // 变更进度条
 ViewGroup.LayoutParams lp = mOperationPercent.getLayoutParams();
 lp.width = findViewById(R.id.operation_full).getLayoutParams().width
 * index / mMaxVolume;
 mOperationPercent.setLayoutParams(lp);
 }

 /**
 * 滑动改变亮度
 *
 * @param percent
 */
 private void onBrightnessSlide(float percent) {
 if (mBrightness < 0) {
 mBrightness = getWindow().getAttributes().screenBrightness;
 if (mBrightness <= 0.00f)
 mBrightness = 0.50f;
 if (mBrightness < 0.01f)
 mBrightness = 0.01f;

 // 显示
 mOperationBg.setImageResource(R.drawable.video_brightness_bg);
 mVolumeBrightnessLayout.setVisibility(View.VISIBLE);
 }
 WindowManager.LayoutParams lpa = getWindow().getAttributes();
 lpa.screenBrightness = mBrightness + percent;
 if (lpa.screenBrightness > 1.0f)
 lpa.screenBrightness = 1.0f;
 else if (lpa.screenBrightness < 0.01f)
 lpa.screenBrightness = 0.01f;
 getWindow().setAttributes(lpa);

 ViewGroup.LayoutParams lp = mOperationPercent.getLayoutParams();
 lp.width = (int) (findViewById(R.id.operation_full).getLayoutParams().width * lpa.screenBrightness);
 mOperationPercent.setLayoutParams(lp);
 }

 @Override
 public void onConfigurationChanged(Configuration newConfig) {
 if (mVideoView != null)
 mVideoView.setVideoLayout(mLayout, 0);
 super.onConfigurationChanged(newConfig);
 }

 } 

3.3 代码说明

3.3.1  缩放功能

该功能SDK已经提供好了接口,直接使用即可。

3.3.2  音量和亮度控制实现

根据layout可以看得出,利用FrameLayout的特点(后面视图会覆盖前面视图),通过控制后一个视图的宽度来达到进度条的效果。

3.3.3  自动隐藏

可用Handle来实现自定延时隐藏的功能,比较实用。

3.3.4  手势

手势方面大家可用多查查GestureDetector方面的资料,双击、缩放手势都可以实现。

 四、代码下载

请移步#Taocode(SVN):(没有账户的请注册一个账户即可。)

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

  五、Vitamio相关信息

5.1 近期将发布新的SDK版本

5.1.1    将直接内置各平台解码器,无需外下载!

5.1.2    将支持自定义进度控制条等。

  六、相关文章

6.1 Android 播放电影时滑动屏幕调整屏幕亮度

6.2 android MediaPlayer API

结束

本系列文章承诺每周至少出一篇,以帮助需要的朋友尽快将Vitamio融合入自己或公司的项目中。

以上就是对Android Vitamio 资料的整理,后续继续补充。

(0)

相关推荐

  • Android中实时获取音量分贝值详解

    基础知识 度量声音强度,大家最熟悉的单位就是分贝(decibel,缩写为dB).这是一个无纲量的相对单位,计算公式如下: 分子是测量值的声压,分母是参考值的声压(20微帕,人类所能听到的最小声压).因此日常中说道声音强度是多少多少分贝时,都是默认了一个很小的参考值的. 而Android设备传感器可以提供的物理量是场的幅值(amplitude),常用下列公式计算分贝值: 从SDK中读取了某段音频数据的振幅后,取最大振幅或平均振幅(可以用平方和平均,或绝对值的和平均),代入上述公式的A1. 现在问题

  • android获取情景模式和铃声 实现震动、铃声提醒

    当我们想通过铃声或者震动提醒用户的时候(类似于手机来电提醒界面),我们需要考虑到手机本身的情景模式.(目前有个OPPO的测试手机就发现,即使调为了静音模式,我依旧可以将铃声播放出来),为了防止"灵异"事件的发生,所以在提示前将情景模式判断以便还是有必要的,特地将代码纪录. 1.获取手机情景模式: AudioManager audioManager = (AudioManager) getSystemService(Context.AUDIO_SERVICE); int ringerMo

  • Android实现音量调节的方法

    本文实例讲述了Android实现音量调节的方法.分享给大家供大家参考.具体如下: main.xml布局文件: <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:

  • Android使用AudioManager修改系统音量的方法

    本文实例讲述了Android使用AudioManager修改系统音量的方法.分享给大家供大家参考,具体如下: 下面介绍几个AudioManager的几个音量调整方面的方法. 首先是得到AudioManager实例: AudioManager am=(AudioManager)getSystemService(Context.AUDIO_SERVICE); 调整音量方法有两种,一种是渐进式,即像手动按音量键一样,一步一步增加或减少,另一种是直接设置音量值. 1.渐进式 public void ad

  • Android编程实现音量按钮添加监听事件的方法

    本文实例讲述了Android编程实现音量按钮添加监听事件的方法.分享给大家供大家参考,具体如下: 很多Android应用都应用到音量按钮,比如翻页,调整音乐声音大小等,但是如果没有对音量按钮进行监听,则无法达到预期的效果.如下代码,就是监听Android手机的音量按钮,开发者可以在相应的位置添加自己需要实现的功能. @Override public boolean onKeyDown (int keyCode, KeyEvent event) { // 获取手机当前音量值 int i = get

  • Android设置铃声实现代码

    本文实例讲述了Android设置铃声实现代码.分享给大家供大家参考.具体如下: public void setMyRingtone(File file) { ContentValues values = new ContentValues(); values.put(MediaStore.MediaColumns.DATA, file.getAbsolutePath()); // values.put(MediaStore.MediaColumns.TITLE, file.getName());

  • Android自定义View 实现闹钟唤起播放闹钟铃声功能

    先上图看一下闹钟唤期页面的效果 实现的功能: 1:转动的图片根据天气情况更换 2:转动时间可以设置,转动结束,闹铃声音就结束 3:光圈颜色渐变效果 直接上代码啦: package com.yuekong.sirius.extension.customview; import android.animation.Animator; import android.animation.ValueAnimator; import android.content.Context; import andro

  • Android给app设置自定义铃声功能

    因为有个需求是在自己的app中添加铃声并播放,而之前在网上查阅了相关的资料却大多都是教人怎么设置手机来电.闹钟等手机系统的功能,所以就以此文章作为记录. 一.本篇文章的大致内容是 1.调用系统铃声界面选择铃声: 2.播放该选择的铃声: 3.再次调用系统铃声界面时,默认勾选之前已经选择过的铃声: 4.非常重要的权限要加上: 二.图示 界面代码很简单,顾名思义. 三.部分逻辑代码实现 ①调用系统铃声界面传递当前铃声数据,并选择新的铃声 //选择铃声 public void toChoose(View

  • Android 自定义SeekBar动态改变硬件音量大小实现和音量键的同步(推荐)

    1,上图: 2,代码: MainActivity.Java package com.hero.zhaoq.seekbarchangeddemo; import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.database.ContentObse

  • Android seekbar(自定义)控制音量同步更新

    Android seekbar控制音量同步更新 作为开发人员来讲,seekbar你一定会碰到,那么怎么自定义seekbar以及seekbar控制的音量怎么与系统的seekbar音量同步更新.今天就下拉菜单中添加SeekBar控制音量为例(Android 5.1系统)   一:自定义Seekbar 先来看一下所自定义的效果图: 首先来来看一下布局文件: <LinearLayout xmlns:android="http://schemas.android.com/apk/res/androi

  • Android中自定义View实现圆环等待及相关的音量调节效果

    圆环交替.等待效果 效果就这样,分析了一下,大概有这几个属性,两个颜色,一个速度,一个圆环的宽度. 自定View的几个步骤: 1.自定义View的属性 2.在View的构造方法中获得我们自定义的属性 3.重写onMesure  4.重写onDraw 1.自定义属性: <?xml version="1.0" encoding="utf-8"?> <resources> <attr name="firstColor" f

随机推荐