Android利用Sensor(传感器)实现水平仪功能

 更新时间:2017年02月27日 10:45:43   作者:欧阳鹏  
这篇文章主要为大家详细介绍了Android利用Sensor传感器实现水平仪功能,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

这里介绍的水平仪,指的是比较传统的气泡水平仪,在一个透明圆盘内充满液体,液体中留有一个气泡,当一端翘起时,该气泡就会浮向翘起的一端。

利用方向传感器返回的第一个参数,实现了一个指南针小应用。

我的Android进阶之旅------>Android利用Sensor(传感器)实现指南针功能

接下来,我们利用返回的第二、三个参数实现该水平仪。因为第二个参数,反映底部翘起的角度(当顶部翘起时为负值),第三个参数可以反映右侧翘起的角度(当左侧翘起时为负值)。根据这两个角度就可以开发水平仪,实现手机哪端翘起,气泡就浮向哪端,这也是水平仪的实现思想。本实例来自于《疯狂Android讲义》

先来看下运行效果:

该程序自定义了一个View,用来绘制透明圆盘和气泡,其中气泡的位置会动态改变。自定义View代码如下:

MyView.java

package org.crazyit.sensor; 
 
import android.content.Context; 
import android.graphics.Bitmap; 
import android.graphics.BitmapFactory; 
import android.graphics.Canvas; 
import android.util.AttributeSet; 
import android.view.View; 
 
public class MyView extends View { 
 // 定义水平仪仪表盘图片 
 Bitmap back; 
 // 定义水平仪中的气泡图标 
 Bitmap bubble; 
 // 定义水平仪中气泡 的X、Y座标 
 int bubbleX, bubbleY; 
 
 public MyView(Context context, AttributeSet attrs) { 
 super(context, attrs); 
 // 加载水平仪图片和气泡图片 
 back = BitmapFactory.decodeResource(getResources(), R.drawable.back); 
 bubble = BitmapFactory 
 .decodeResource(getResources(), R.drawable.bubble); 
 } 
 
 @Override 
 protected void onDraw(Canvas canvas) { 
 super.onDraw(canvas); 
 // 绘制水平仪表盘图片 
 canvas.drawBitmap(back, 0, 0, null); 
 // 根据气泡座标绘制气泡 
 canvas.drawBitmap(bubble, bubbleX, bubbleY, null); 
 } 
} 

布局文件 main.xml

<?xml version="1.0" encoding="utf-8"?> 
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" 
 android:orientation="vertical" 
 android:layout_width="fill_parent" 
 android:layout_height="fill_parent" 
 android:background="#fff" 
 > 
<org.crazyit.sensor.MyView 
 android:id="@+id/show" 
 android:layout_width="fill_parent" 
 android:layout_height="fill_parent" 
/> 
</FrameLayout> 

素材:

bubble.png:

back.png :

Gradienter.java

package org.crazyit.sensor; 
 
import android.app.Activity; 
import android.hardware.Sensor; 
import android.hardware.SensorEvent; 
import android.hardware.SensorEventListener; 
import android.hardware.SensorManager; 
import android.os.Bundle; 
 
public class Gradienter extends Activity implements SensorEventListener { 
 // 定义水平仪的仪表盘 
 MyView show; 
 // 定义水平仪能处理的最大倾斜角,超过该角度,气泡将直接在位于边界。 
 int MAX_ANGLE = 30; 
 // 定义Sensor管理器 
 SensorManager mSensorManager; 
 
 @Override 
 public void onCreate(Bundle savedInstanceState) { 
 super.onCreate(savedInstanceState); 
 setContentView(R.layout.main); 
 // 获取水平仪的主组件 
 show = (MyView) findViewById(R.id.show); 
 // 获取传感器管理服务 
 mSensorManager = (SensorManager) getSystemService(SENSOR_SERVICE); 
 } 
 
 @Override 
 public void onResume() { 
 super.onResume(); 
 // 为系统的方向传感器注册监听器 
 mSensorManager.registerListener(this, 
 mSensorManager.getDefaultSensor(Sensor.TYPE_ORIENTATION), 
 SensorManager.SENSOR_DELAY_GAME); 
 } 
 
 @Override 
 protected void onPause() { 
 // 取消注册 
 mSensorManager.unregisterListener(this); 
 super.onPause(); 
 } 
 
 @Override 
 protected void onStop() { 
 // 取消注册 
 mSensorManager.unregisterListener(this); 
 super.onStop(); 
 } 
 
 @Override 
 public void onAccuracyChanged(Sensor sensor, int accuracy) { 
 } 
 
 @Override 
 public void onSensorChanged(SensorEvent event) { 
 float[] values = event.values; 
 // 获取触发event的传感器类型 
 int sensorType = event.sensor.getType(); 
 switch (sensorType) { 
 case Sensor.TYPE_ORIENTATION: 
 // 获取与Y轴的夹角 
 float yAngle = values[1]; 
 // 获取与Z轴的夹角 
 float zAngle = values[2]; 
 // 气泡位于中间时(水平仪完全水平),气泡的X、Y座标 
 int x = (show.back.getWidth() - show.bubble.getWidth()) / 2; 
 int y = (show.back.getHeight() - show.bubble.getHeight()) / 2; 
 // 如果与Z轴的倾斜角还在最大角度之内 
 if (Math.abs(zAngle) <= MAX_ANGLE) { 
 // 根据与Z轴的倾斜角度计算X座标的变化值(倾斜角度越大,X座标变化越大) 
 int deltaX = (int) ((show.back.getWidth() - show.bubble 
  .getWidth()) / 2 * zAngle / MAX_ANGLE); 
 x += deltaX; 
 } 
 // 如果与Z轴的倾斜角已经大于MAX_ANGLE,气泡应到最左边 
 else if (zAngle > MAX_ANGLE) { 
 x = 0; 
 } 
 // 如果与Z轴的倾斜角已经小于负的MAX_ANGLE,气泡应到最右边 
 else { 
 x = show.back.getWidth() - show.bubble.getWidth(); 
 } 
 // 如果与Y轴的倾斜角还在最大角度之内 
 if (Math.abs(yAngle) <= MAX_ANGLE) { 
 // 根据与Y轴的倾斜角度计算Y座标的变化值(倾斜角度越大,Y座标变化越大) 
 int deltaY = (int) ((show.back.getHeight() - show.bubble 
  .getHeight()) / 2 * yAngle / MAX_ANGLE); 
 y += deltaY; 
 } 
 // 如果与Y轴的倾斜角已经大于MAX_ANGLE,气泡应到最下边 
 else if (yAngle > MAX_ANGLE) { 
 y = show.back.getHeight() - show.bubble.getHeight(); 
 } 
 // 如果与Y轴的倾斜角已经小于负的MAX_ANGLE,气泡应到最右边 
 else { 
 y = 0; 
 } 
 // 如果计算出来的X、Y座标还位于水平仪的仪表盘内,更新水平仪的气泡座标 
 if (isContain(x, y)) { 
 show.bubbleX = x; 
 show.bubbleY = y; 
 } 
 // 通知系统重回MyView组件 
 show.postInvalidate(); 
 break; 
 } 
 } 
 
 // 计算x、y点的气泡是否处于水平仪的仪表盘内 
 private boolean isContain(int x, int y) { 
 // 计算气泡的圆心座标X、Y 
 int bubbleCx = x + show.bubble.getWidth() / 2; 
 int bubbleCy = y + show.bubble.getWidth() / 2; 
 // 计算水平仪仪表盘的圆心座标X、Y 
 int backCx = show.back.getWidth() / 2; 
 int backCy = show.back.getWidth() / 2; 
 // 计算气泡的圆心与水平仪仪表盘的圆心之间的距离。 
 double distance = Math.sqrt((bubbleCx - backCx) * (bubbleCx - backCx) 
 + (bubbleCy - backCy) * (bubbleCy - backCy)); 
 // 若两个圆心的距离小于它们的半径差,即可认为处于该点的气泡依然位于仪表盘内 
 if (distance < (show.back.getWidth() - show.bubble.getWidth()) / 2) { 
 return true; 
 } else { 
 return false; 
 } 
 } 
} 

AndroidManifest.xml

<?xml version="1.0" encoding="utf-8"?> 
<manifest xmlns:android="http://schemas.android.com/apk/res/android" 
 package="org.crazyit.sensor" 
 android:versionCode="1" 
 android:versionName="1.0"> 
 <uses-sdk 
 android:minSdkVersion="10" 
 android:targetSdkVersion="17" /> 
 <application android:icon="@drawable/ic_launcher" android:label="@string/app_name"> 
 <activity android:name=".Gradienter" 
 android:label="@string/app_name"> 
 <intent-filter> 
 <action android:name="android.intent.action.MAIN" /> 
 <category android:name="android.intent.category.LAUNCHER" /> 
 </intent-filter> 
 </activity> 
 </application> 
</manifest> 

PS:请在真机环境下运行此程序,如果在模拟器下运行,可能没效果。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家

相关文章

  • Android导航栏功能项的显示与屏蔽介绍

    Android导航栏功能项的显示与屏蔽介绍

    大家好,本篇文章主要讲的是Android导航栏功能项的显示与屏蔽介绍,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下,方便下次浏览
    2021-12-12
  • Flutter调用Android和iOS原生代码的方法示例

    Flutter调用Android和iOS原生代码的方法示例

    这篇文章主要给大家介绍了关于Flutter调用Android和iOS原生代码的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用Flutter具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-04-04
  • Android中检测当前是否为主线程最可靠的解决方法

    Android中检测当前是否为主线程最可靠的解决方法

    这篇文章主要介绍了Android中检测当前是否为主线程最可靠的解决方法,本文先是给出了最可靠的方法,然后给出了几个实验例子,需要的朋友可以参考下
    2015-01-01
  • Android Tab 控件详解及实例

    Android Tab 控件详解及实例

    这篇文章主要介绍了Android Tab 控件详解及实例的相关资料,需要的朋友可以参考下
    2017-03-03
  • Kotlin fun函数使用方法

    Kotlin fun函数使用方法

    函数是执行特定任务的一组相互关联的代码块。函数用于将程序分解为不同的子模块。它使代码可重用,并使程序更易于管理,这篇文章主要介绍了Kotlin fun函数使用方法
    2022-12-12
  • Android唤醒、解锁屏幕代码实例

    Android唤醒、解锁屏幕代码实例

    这篇文章主要介绍了Android唤醒、解锁屏幕代码实例,本文讲解了唤醒、解锁屏幕需要的权限和操作代码实例,代码中包含详细注释,需要的朋友可以参考下
    2015-05-05
  • Android编程监听APK安装与删除等过程的方法

    Android编程监听APK安装与删除等过程的方法

    这篇文章主要介绍了Android编程监听APK安装与删除等过程的方法,涉及Android事件监听、权限控制、广播操作等相关实现技巧,需要的朋友可以参考下
    2017-10-10
  • 深入解读Android的内部进程通信接口AIDL

    深入解读Android的内部进程通信接口AIDL

    这篇文章主要介绍了Android的内部进程通信接口AIDL,重点讲解了进程间的通信与AIDL内存使用方面的parcelable接口的实现,需要的朋友可以参考下
    2016-04-04
  • 30条android项目开发技巧与经验总结

    30条android项目开发技巧与经验总结

    本文为大家总结了30条android项目开发技巧与经验,,需要的朋友可以参考下
    2018-04-04
  • Android学习教程之悬浮窗菜单制作(9)

    Android学习教程之悬浮窗菜单制作(9)

    这篇文章主要为大家详细介绍了Android学习教程之悬浮窗菜单制作方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-11-11

最新评论