欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

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

程序员文章站 2023-12-13 21:27:10
这里介绍的水平仪,指的是比较传统的气泡水平仪,在一个透明圆盘内充满液体,液体中留有一个气泡,当一端翘起时,该气泡就会浮向翘起的一端。 利用方向传感器返回的第一个参数,...

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

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

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

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

先来看下运行效果:

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

该程序自定义了一个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:Android利用Sensor(传感器)实现水平仪功能

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

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:请在真机环境下运行此程序,如果在模拟器下运行,可能没效果。

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

上一篇:

下一篇: