【发布时间】:2012-01-09 02:19:24
【问题描述】:
我是 Android 新手,我想根据我的相机获得方向。如何根据我的相机获取方向信息?你能给出一个想法吗?
【问题讨论】:
标签: android android-manifest compass-geolocation
我是 Android 新手,我想根据我的相机获得方向。如何根据我的相机获取方向信息?你能给出一个想法吗?
【问题讨论】:
标签: android android-manifest compass-geolocation
我们不能再使用方向传感器了,我们可以同时使用磁场传感器和加速度传感器来获得同等的功能。它的工作量更大,但它确实允许继续使用回调来处理方向变化。
从加速度计和磁场到方位角的转换:
float[] mGravity;
float[] mGeomagnetic;
public void onSensorChanged(SensorEvent event) {
if (event.sensor.getType() == Sensor.TYPE_ACCELEROMETER)
mGravity = event.values;
if (event.sensor.getType() == Sensor.TYPE_MAGNETIC_FIELD)
mGeomagnetic = event.values;
if (mGravity != null && mGeomagnetic != null) {
float R[] = new float[9];
float I[] = new float[9];
if (SensorManager.getRotationMatrix(R, I, mGravity, mGeomagnetic)) {
// orientation contains azimut, pitch and roll
float orientation[] = new float[3];
SensorManager.getOrientation(R, orientation);
azimut = orientation[0];
}
}
}
要指向北方,您可以计算旋转度数:
float rotation = -azimut * 360 / (2 * 3.14159f);
【讨论】:
到目前为止,这对我来说有点用,返回的值在 0 到 360 之间,但我认为北方没有正确校准?我在运行 Android 5.0.1 的 LG G Pad 上使用它
public void onSensorChanged(SensorEvent event) {
if (event.sensor.getType() == Sensor.TYPE_ACCELEROMETER)
mGravity = event.values;
if (event.sensor.getType() == Sensor.TYPE_MAGNETIC_FIELD)
mGeomagnetic = event.values;
if (mGravity != null && mGeomagnetic != null) {
float R[] = new float[9];
float outR[] = new float[9];
float I[] = new float[9];
boolean success = SensorManager.getRotationMatrix(R, I, mGravity, mGeomagnetic);
if (success) {
float orientation[] = new float[3];
SensorManager.remapCoordinateSystem(R, SensorManager.AXIS_X, SensorManager.AXIS_Y, outR);
SensorManager.getOrientation(outR, orientation);
azimut = orientation[0];
float degree = (float)(Math.toDegrees(azimut)+360)%360;
System.out.println("degree " + degree);
我确信我错过了一些东西,但希望这对其他人来说是一个不错的起点。为了达到这一点,我查看了很多其他问题、cmets 等。
【讨论】:
【讨论】:
使用结合了地磁和重力传感器的旋转矢量传感器。然后使用SensorManager.getOrientation 将 5 个值的数组转换为 OpenGL 样式的旋转矩阵。正如文档所说,第一个值是以弧度表示的指南针方向。
这本身并不能解决您从相机的角度了解指南针方向的问题。相反,它假设屏幕与地面平行,就像老式的袖珍指南针一样,它会报告屏幕顶部指向的方向。因此,如果屏幕顶部朝北,则方向 (event.values[0]) 为 0。如果屏幕顶部笔直向上,则方向未定义。不幸的是,这在使用相机时很常见。
在示例监听器中,我将使用枚举在 Android 的默认袖珍罗盘风格方向和后置摄像头方向之间切换,这样您就可以看到 Android 期望的用例和您想要的用例。
enum class CompassCoordinateSystem { POCKET_COMPASS, REAR_CAMERA }
然后写一个SensorEventListener 来跟踪更改。为了清楚起见,我没有在下面的代码中import android.hardware.SensorManager.*。下面定义的所有数组都由 SensorManager 静态方法填充。
/** The latest compass orientation as a 3D vector. */
private var orientation3D = FloatArray(3)
private var coordinateSystem = CompassCoordinateSystem.REAR_CAMERA_ROTATION
fun compassDegrees(): Float = azimuthToDegrees(compassRadians())
fun compassRadians(): Float = orientation3D[0]
/** Convert such that North=0, East=90, South=180, West=270. */
fun azimuthToDegrees(azimuth: Float): Float {
return ((Math.toDegrees(azimuth.toDouble())+360) % 360).toFloat()
}
override fun onSensorChanged(event: SensorEvent?) {
if (event?.sensor?.type == Sensor.TYPE_ROTATION_VECTOR) {
val rotationMatrix = FloatArray(9)
SensorManager.getRotationMatrixFromVector(rotationMatrix, event.values)
when (coordinateSystem) {
CompassCoordinateSystem.POCKET_COMPASS -> SensorManager.getOrientation(rotationMatrix, orientation3D)
CompassCoordinateSystem.REAR_CAMERA -> {
val rearCameraMatrix = FloatArray(9)
// The axis parameters for remapCoordinateSystem() are
// from an example in that method's documentation
SensorManager.remapCoordinateSystem(rotationMatrix,
SensorManager.AXIS_X,
SensorManager.AXIS_Z,
rearCameraMatrix)
SensorManager.getOrientation(rearCameraMatrix, orientation3D)
}
}
}
}
【讨论】: