Android系统带的传感器有很多种,最常见的莫过于微信的摇一摇了,那么今天我们就来看看Anroid中传感器的使用,做一个类似于微信摇一摇的效果。

OK ,废话不多说,我们就先来看看效果图吧:

当我摇动手机的时候这里的动画效果基本和微信上的动画效果一致,这里请大家自行脑补微信摇一摇画面。

那我们就动手吧。

1.布局文件

好,那我们先来看看布局文件吧,在布局文件的正中央是一个花的图片,上图大家看到的手机图片实际上是两张图片拼接在一起,将花的那张图片遮住了,当摇一摇的时候,这两张图片分别向上或者向下移动,然后花的图片就可以显示出来。OK,基本原理就是这样,我们来看看代码:

[java] view plaincopy print?
  1. <?xml version="1.0" encoding="utf-8"?>
  2. <RelativeLayout
  3. xmlns:android="http://schemas.android.com/apk/res/android"
  4. xmlns:tools="http://schemas.android.com/tools"
  5. android:layout_width="match_parent"
  6. android:layout_height="match_parent"
  7. android:background="#1f1f1f">
  8. <ImageView
  9. android:layout_width="wrap_content"
  10. android:layout_height="wrap_content"
  11. android:src="@drawable/flower"/>
  12. <LinearLayout
  13. android:layout_width="wrap_content"
  14. android:layout_height="wrap_content"
  15. android:layout_centerInParent="true"
  16. android:orientation="vertical">
  17. <ImageView
  18. android:id="@+id/up"
  19. android:layout_width="wrap_content"
  20. android:layout_height="wrap_content"
  21. android:src="@drawable/up"/>
  22. <ImageView
  23. android:id="@+id/down"
  24. android:layout_width="wrap_content"
  25. android:layout_height="wrap_content"
  26. android:src="@drawable/down"/>
  27. </LinearLayout>
  28. </RelativeLayout>

2.传感器监听手机晃动

既然要监听手机加速度的变化,那我首先需要获取系统的传感器:

[java] view plaincopy print?
  1. //获取到一个传感器管理器
  2. sensorManager = (SensorManager) getSystemService(SENSOR_SERVICE);
  3. //获得一个加速度传感器
  4. Sensor sensor = sensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);

这两行代码首先是获取一个传感器管理器,然后获取加速度传感器,因为关于传感器的API 有很多,这里你需要指明自己要获取的是哪一个传感器。拿到传感器之后,需要注册监听,如下:

[java] view plaincopy print?
  1. sensorManager.registerListener(listener, sensor, SensorManager.SENSOR_DELAY_GAME);

注册传感器的监听一共需要三个参数,第一个是监听器,第二个是加速度传感器,第三个是传感器的灵敏度,传感器的灵敏度一共分为四级,如下,从上往下灵敏度依次降低:

    1. SENSOR_DELAY_FASTEST
   2. SENSOR_DELAY_GAME
   3. SENSOR_DELAY_UI
   4. SENSOR_DELAY_NORMAL

OK ,注册完之后,我们还是来看看这个监听器是什么吧:

[java] view plaincopy print?
  1. private SensorEventListener listener = new SensorEventListener() {
  2. //当手机的加速度发生变化时调用
  3. @Override
  4. public void onSensorChanged(SensorEvent event) {
  5. //获取手机在不同方向上加速度的变化
  6. float valuesX = Math.abs(event.values[0]);
  7. float valuesY = Math.abs(event.values[1]);
  8. float valuesZ = Math.abs(event.values[2]);
  9. if (valuesX > 17 || valuesY > 17 || valuesZ > 17) {
  10. startAnimation();
  11. playSound();
  12. }
  13. }
  14. @Override
  15. public void onAccuracyChanged(Sensor sensor, int accuracy) {
  16. }
  17. };

这个listener中一共就两个方法,一个是当手机的加速度发生改变的时候调用,还有一个是当传感器的灵敏度发生改变的时候调用,当手机的加速度发生改变的时候,我们可以获取到手机在X 、Y、Z 三个维度上的变化值,拿到这个值之后,我们只需要进行简单的比较即可,如果有任意一个方向的值大于17,则认为有人在晃动手机,这个时候开启动画和声音的播放。

3.开启动画和声音

动画实际上就是两个平移动画,我们来看看:

[java] view plaincopy print?
  1. private void startAnimation() {
  2. //如果两次晃动手机的时间小于1秒,则只执行一次动画
  3. long currentTimeMillis = System.currentTimeMillis();
  4. if (currentTimeMillis - lastTime < 1000) {
  5. return;
  6. }
  7. lastTime = currentTimeMillis;
  8. AnimationSet upSet = new AnimationSet(true);
  9. TranslateAnimation upUp = new TranslateAnimation(TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF,
  10. 0, TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF, -1);
  11. upUp.setDuration(1000);
  12. TranslateAnimation upDown = new TranslateAnimation(TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF,
  13. 0, TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF, 1);
  14. upDown.setDuration(1000);
  15. upDown.setStartOffset(1000);
  16. upSet.addAnimation(upUp);
  17. upSet.addAnimation(upDown);
  18. up.startAnimation(upSet);
  19. AnimationSet downSet = new AnimationSet(true);
  20. TranslateAnimation downUp = new TranslateAnimation(TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF,
  21. 0, TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF, 1);
  22. downUp.setDuration(1000);
  23. TranslateAnimation downDown = new TranslateAnimation(TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF,
  24. 0, TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF, -1);
  25. downDown.setDuration(1000);
  26. downDown.setStartOffset(1000);
  27. downSet.addAnimation(downUp);
  28. downSet.addAnimation(downDown);
  29. down.startAnimation(downSet);
  30. }

至于声音,由于我这里只是播放比较短小的音效而已,所以并没有必要使用MediaPlayer,我可以通过一个声音池来解决这个问题,代码如下:

[java] view plaincopy print?
  1. /**
  2. * 初始化声音池
  3. */
  4. private void initSoundPool() {
  5. if (Build.VERSION.SDK_INT > 20) {
  6. SoundPool.Builder builder = new SoundPool.Builder();
  7. //1.最大并发流数
  8. builder.setMaxStreams(3);
  9. AudioAttributes.Builder aaBuilder = new AudioAttributes.Builder();
  10. aaBuilder.setLegacyStreamType(AudioManager.STREAM_MUSIC);
  11. builder.setAudioAttributes(aaBuilder.build());
  12. soundPool = builder.build();
  13. } else {
  14. soundPool = new SoundPool(3, AudioManager.STREAM_MUSIC, 0);
  15. }
  16. //加载一个音频文件
  17. sound1 = soundPool.load(this, R.raw.awe, 1);
  18. }

在创建一个声音池的时候我采取了两种不同的方案,如果系统的版本大于20,则是用第一种方式获取声音池,否则使用第二种方式获取声音池。获取声音池之后,再通过声音池加载一个音频文件。加载完成之后,我就可以对这个音频文件进行播放了,如下:

[java] view plaincopy print?
  1. //1.声音的id
  2. //2.3.表示左右声道的音量
  3. //4.优先级
  4. //5.是否循环
  5. //6.声音播放速率
  6. soundPool.play(sound1, 1, 1, 0, 0, 1);

每个参数的含义都写的很清楚了,大家又不清楚的地方可以直接看源码,这里的源码注释很好懂。

最后一步就是开启手机震动了,开启手机震动,我需要首先获取震动服务,如下:

[java] view plaincopy print?
  1. //获取手机震动服务
  2. vibrator = (Vibrator) getSystemService(VIBRATOR_SERVICE);

然后调用Vibrator类中的vibrator方法执行震动,如下:

[java] view plaincopy print?
  1. //1.表示震动的节奏off/on/off/on/off/on......
  2. //2.表示是否重复震动,-1表示不重复
  3. vibrator.vibrate(new long[]{100, 200, 100, 200, 100, 200}, -1);

手机震动一定要记得添加震动权限哦,如下:

[java] view plaincopy print?
  1. <uses-permission android:name="android.permission.VIBRATE" />

OK ,最后,在销毁Activity的时候要解除对传感器的监听,同时释放声音池资源,如下:

[java] view plaincopy print?
  1. @Override
  2. protected void onDestroy() {
  3. super.onDestroy();
  4. //解除对加速度传感器的监听
  5. sensorManager.unregisterListener(listener);
  6. if (soundPool != null) {
  7. //声音池释放资源
  8. soundPool.release();
  9. }
  10. }

完整的Activity 代码如下:

[java] view plaincopy print?
  1. public class MainActivity extends AppCompatActivity {
  2. private ImageView up;
  3. private ImageView down;
  4. //上一次晃动手机的时间
  5. private long lastTime;
  6. private SoundPool soundPool;
  7. private int sound1;
  8. private Vibrator vibrator;
  9. private SensorEventListener listener = new SensorEventListener() {
  10. //当手机的加速度发生变化时调用
  11. @Override
  12. public void onSensorChanged(SensorEvent event) {
  13. //获取手机在不同方向上加速度的变化
  14. float valuesX = Math.abs(event.values[0]);
  15. float valuesY = Math.abs(event.values[1]);
  16. float valuesZ = Math.abs(event.values[2]);
  17. if (valuesX > 17 || valuesY > 17 || valuesZ > 17) {
  18. startAnimation();
  19. playSound();
  20. }
  21. }
  22. @Override
  23. public void onAccuracyChanged(Sensor sensor, int accuracy) {
  24. }
  25. };
  26. private SensorManager sensorManager;
  27. private void playSound() {
  28. //1.声音的id
  29. //2.3.表示左右声道的音量
  30. //4.优先级
  31. //5.是否循环
  32. //6.声音播放速率
  33. soundPool.play(sound1, 1, 1, 0, 0, 1);
  34. //手机震动
  35. //1.表示震动的节奏off/on/off/on/off/on......
  36. //2.表示是否重复震动,-1表示不重复
  37. vibrator.vibrate(new long[]{100, 200, 100, 200, 100, 200}, -1);
  38. }
  39. private void startAnimation() {
  40. //如果两次晃动手机的时间小于1秒,则只执行一次动画
  41. long currentTimeMillis = System.currentTimeMillis();
  42. if (currentTimeMillis - lastTime < 1000) {
  43. return;
  44. }
  45. lastTime = currentTimeMillis;
  46. AnimationSet upSet = new AnimationSet(true);
  47. TranslateAnimation upUp = new TranslateAnimation(TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF,
  48. 0, TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF, -1);
  49. upUp.setDuration(1000);
  50. TranslateAnimation upDown = new TranslateAnimation(TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF,
  51. 0, TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF, 1);
  52. upDown.setDuration(1000);
  53. upDown.setStartOffset(1000);
  54. upSet.addAnimation(upUp);
  55. upSet.addAnimation(upDown);
  56. up.startAnimation(upSet);
  57. AnimationSet downSet = new AnimationSet(true);
  58. TranslateAnimation downUp = new TranslateAnimation(TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF,
  59. 0, TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF, 1);
  60. downUp.setDuration(1000);
  61. TranslateAnimation downDown = new TranslateAnimation(TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF,
  62. 0, TranslateAnimation.RELATIVE_TO_SELF, 0, TranslateAnimation.RELATIVE_TO_SELF, -1);
  63. downDown.setDuration(1000);
  64. downDown.setStartOffset(1000);
  65. downSet.addAnimation(downUp);
  66. downSet.addAnimation(downDown);
  67. down.startAnimation(downSet);
  68. }
  69. @Override
  70. protected void onCreate(Bundle savedInstanceState) {
  71. super.onCreate(savedInstanceState);
  72. setContentView(R.layout.activity_main);
  73. up = ((ImageView) findViewById(R.id.up));
  74. down = ((ImageView) findViewById(R.id.down));
  75. initSensor();
  76. initSoundPool();
  77. //获取手机震动服务
  78. vibrator = (Vibrator) getSystemService(VIBRATOR_SERVICE);
  79. }
  80. /**
  81. * 初始化声音池
  82. */
  83. private void initSoundPool() {
  84. if (Build.VERSION.SDK_INT > 20) {
  85. SoundPool.Builder builder = new SoundPool.Builder();
  86. //1.最大并发流数
  87. builder.setMaxStreams(3);
  88. AudioAttributes.Builder aaBuilder = new AudioAttributes.Builder();
  89. aaBuilder.setLegacyStreamType(AudioManager.STREAM_MUSIC);
  90. builder.setAudioAttributes(aaBuilder.build());
  91. soundPool = builder.build();
  92. } else {
  93. soundPool = new SoundPool(3, AudioManager.STREAM_MUSIC, 0);
  94. }
  95. //加载一个音频文件
  96. sound1 = soundPool.load(this, R.raw.awe, 1);
  97. }
  98. /**
  99. * 初始化传感器
  100. */
  101. private void initSensor() {
  102. //获取到一个传感器管理器
  103. sensorManager = (SensorManager) getSystemService(SENSOR_SERVICE);
  104. //获得一个加速度传感器
  105. Sensor sensor = sensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);
  106. //注册传感器监听,
  107. //1.监听器
  108. //2.加速度传感器
  109. //3.传感器灵敏度
  110. //传感器灵敏度分为四级,从上往下灵敏度依次降低
  111. //SENSOR_DELAY_FASTEST
  112. //SENSOR_DELAY_GAME
  113. //SENSOR_DELAY_UI
  114. //SENSOR_DELAY_NORMAL
  115. sensorManager.registerListener(listener, sensor, SensorManager.SENSOR_DELAY_GAME);
  116. }
  117. @Override
  118. protected void onDestroy() {
  119. super.onDestroy();
  120. //解除对加速度传感器的监听
  121. sensorManager.unregisterListener(listener);
  122. if (soundPool != null) {
  123. //声音池释放资源
  124. soundPool.release();
  125. }
  126. }
  127. }

以上。

转载于:https://www.cnblogs.com/Free-Thinker/p/6544174.html

玩转Android之加速度传感器的使用,模仿微信摇一摇相关推荐

  1. android 获取加速度传感器值,Android开发获取传感器数据的方法示例【加速度传感器,磁场传感器,光线传感器,方向传感器】...

    本文实例讲述了Android开发获取传感器数据的方法.分享给大家供大家参考,具体如下: package mobile.android.sensor; import java.util.List; im ...

  2. Android 使用加速度传感器实现摇一摇功能及优化

    如有转载,请声明出处: 时之沙: http://blog.csdn.net/t12x3456 目前很多应用已经实现了摇一摇功能,这里通过讲解该功能的原理及实现回顾一下加速度传感器的使用: 1.首先获得 ...

  3. android 手机加速度传感器的问题

    我自己的手机是华为荣耀3c的,在调用加速度传感器的时候,根据3个方向的加速度判断,然后如果符合则引发震动等效果代码: @Override public void onSensorChanged(Sen ...

  4. android 通过加速度传感器实现手机来电时翻转静音

    实现思路 通过加速度传感器获取 X,Y,Z 三轴所受的重力 计算在指定的时间段内,手机在 X,Y,Z 矢量和的方向上的速度 如果速度大于指定的速度阀值,判定手机当前在剧烈抖动:并结束当前方法 通过Z轴 ...

  5. wing带你玩转自定义view系列(3)模仿微信下拉眼睛

    发现了爱神的自定义view系列,我只想说一个字:凸(艹皿艹 ) !!相见恨晚啊,早看到就不会走这么多弯路了 另外相比之下我这完全是小儿科..所以不说了,这篇是本系列完结篇....我要从零开始跟随爱哥脚 ...

  6. Android加速度传感器

    Android加速度传感器 效果图 手机平放桌面的两张截屏,数据一直在刷新 源码 下载地址(Android Studio工程):http://download.csdn.net/detail/q487 ...

  7. Android 加速度传感器(G-Sensor)

    Android 加速度传感器的类型是  Sensor.TYPE_ACCELEROMETER 通过 android.hardware.SensorEvent 返回加速度传感器值. 加速度传感器返回值的单 ...

  8. Android Sensor Orientation 传感器开发

    手机方向图 将手机平放在桌面上来看,手机的左下角是坐标原点,水平向右为x轴,水平向前为y轴,由x轴和y轴正方向叉乘形成的方向 为z轴,当手机平放时,z轴指向天空的方向. 注释: 手机放平: 水平旋转 ...

  9. Android移动开发-利用加速度传感器开发仿微信摇一摇功能的实现

    加速度传感器是最常见的传感器,大部分Android手机都内置了加速度传感器,加速度传感器运用最广泛的功能就是微信的摇一摇功能,用户通过摇晃手机寻找周围的人,其它类似的应用还摇骰子.玩游戏等. 下面以摇 ...

最新文章

  1. 生成对抗网络GAN综述
  2. vim 的 grep 插件`Leaderf rg`:grep 和模糊匹配的完美结合
  3. python学习并发编程
  4. 多个app用同一个签名文件_手机APP和手机网站的区别有哪些?
  5. 转 Android的Activity屏幕切换动画(一)-左右滑动切换
  6. java学习(160):interrupt方法
  7. Java并发编程系列
  8. 梅森旋转产生随机数c语言实现,C++生成随机数的实现代码
  9. flutter 的gradle下载不了怎么办
  10. k8s学习:部署 PHP + Redis 留言薄
  11. 6月8日 论文书写——公式
  12. 人工智能(12)大数据
  13. 便携式电热水壶外贸出口欧洲CE认证准备资料
  14. 56: Recv failure: Connection was reset和55错误解决办法
  15. 直击AI场景化实践:在部分行业可取代人工
  16. 用计算机做电子贺卡送祝福,全国信息技术优质课一等奖电子贺卡送祝福教案.doc...
  17. 东南大学计算机程光,东南大学计算机科学与工程学院硕导介绍:程光
  18. 深度相机(八)--OpenNI及与Kinect for windows SDK的比较
  19. 量子技术到底是什么?
  20. 云计算技术及其应用前景分析

热门文章

  1. 配置防火墙打开 80 端口
  2. 2015年 六·一 儿童节——我
  3. 关于ecshop中jquery与js冲突解决的方案
  4. 实现对gridview删除行时弹出确认对话框的四种方法
  5. 编译错误syntax error : missing ';' before 'type'原因探寻
  6. ArcGIS编辑操作的常用快捷键一览表
  7. jracdrive变频器说明书580_jracdrive变频器err02
  8. Matlab矩阵的拼接
  9. docker-compose安装问题
  10. 牛客网-数据结构笔试题目(二)-万万没想到之抓捕孔连顺思路解析(附源码)