Android利用Sensor(传感器)实现水平仪功能
程序员文章站
2023-12-17 15:44:16
这里介绍的水平仪,指的是比较传统的气泡水平仪,在一个透明圆盘内充满液体,液体中留有一个气泡,当一端翘起时,该气泡就会浮向翘起的一端。
利用方向传感器返回的第一个参数,...
这里介绍的水平仪,指的是比较传统的气泡水平仪,在一个透明圆盘内充满液体,液体中留有一个气泡,当一端翘起时,该气泡就会浮向翘起的一端。
利用方向传感器返回的第一个参数,实现了一个指南针小应用。
我的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:请在真机环境下运行此程序,如果在模拟器下运行,可能没效果。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持
推荐阅读