对于我正在制作的应用程序,我需要一个摄像头和一个指南针。应用程序在清单中设置为横向模式
首先我实施了指南针。正如Android开发者中所建议的那样,我使用了两个传感器 - Accelerometer 和 Magnetic Field 。我就是这样做的:
我有我的活动工具SensorEventListener
。在onCreate()
我使用:
sensorManager
sManager = (SensorManager) getSystemService(SENSOR_SERVICE);
我在onResume()
注册我的听众:
sManager.registerListener(this, sManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER),SensorManager.SENSOR_DELAY_NORMAL);
sManager.registerListener(this, sManager.getDefaultSensor(Sensor.TYPE_MAGNETIC_FIELD),SensorManager.SENSOR_DELAY_NORMAL);
当然会在onPause()
中取消注册。
我不使用onAccuracyChanged()
。这就是我在onSensorChanged()
中所做的:
@Override
public void onSensorChanged(SensorEvent event) {
switch (event.sensor.getType()) {
case Sensor.TYPE_MAGNETIC_FIELD:
mags = event.values.clone();
break;
case Sensor.TYPE_ACCELEROMETER:
accels = event.values.clone();
break;
}
if (mags != null && accels != null) {
gravity = new float[9];
magnetic = new float[9];
SensorManager.getRotationMatrix(gravity, magnetic, accels, mags);
float[] outGravity = new float[9];
float inclination = (float) Math.acos(gravity[8]);
if (inclination < Math.toRadians(25)
|| inclination > Math.toRadians(155)) {
// device is close to flat. Remap for landscape.
SensorManager.remapCoordinateSystem(gravity, SensorManager.AXIS_Y,SensorManager.AXIS_MINUS_X, outGravity);
SensorManager.getOrientation(outGravity, values);
} else {
// device is not flat. Remap for landscape and perpendicular
SensorManager.remapCoordinateSystem(gravity, SensorManager.AXIS_X,SensorManager.AXIS_Z, outGravity);
SensorManager.getOrientation(outGravity, values);
}
azimuth = Math.round(Math.toDegrees(values[0]));
}
}
正如您所看到的,我区分了手机平躺在桌面上以及用户握住手机时(就像拍照时一样)。当我单独使用这个代码时,一切都或多或少都有效。我正在获得正确的方位角值,当手机躺在桌子上和垂直于桌子时(大约5-10度的差异,但我可以忍受)。
将相机预览添加到应用程序时,问题就开始了。
我有我的活动工具SurfaceHolder.Callback
。我在onCreate()
中初始化了相机:
SurfaceView cameraView = (SurfaceView)findViewById(R.id.camera_view);
surfaceHolder = cameraView.getHolder();
surfaceHolder.addCallback(this);
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.HONEYCOMB) {
surfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
}
这是我实现界面的方式:
@Override
public void surfaceCreated(SurfaceHolder surfaceHolder) {
camera = Camera.open();
camera.setDisplayOrientation(0);
}
@Override
public void surfaceChanged(SurfaceHolder surfaceHolder, int i, int i1, int i2) {
if (isCameraOn) {
camera.stopPreview();
isCameraOn = false;
}
if (camera != null) {
try {
camera.setPreviewDisplay(surfaceHolder);
camera.startPreview();
isCameraOn = true;
} catch (IOException e) {
e.printStackTrace();
}
}
}
@Override
public void surfaceDestroyed(SurfaceHolder surfaceHolder) {
camera.stopPreview();
camera.release();
camera = null;
}
当我将相机代码添加到我的项目中,并在手机屏幕上显示相机时,当手机突然垂直时,我的传感器无法正常工作。如果手机平躺在桌子上,我得到的方位角值是正确的。当手机垂直于桌子时,我的方位角值偏差约40度(虽然稳定)
我已经尝试过寻找解决方案(无论是我自己还是在线),但到目前为止,我的努力都是徒劳的。我想知道如何解决这个问题。
谢谢!
答案 0 :(得分:0)
首个TYPE_MAGNETIC_FIELD传感器不适用于所有设备。 您可以单独使用TYPE_ACCELEROMETER传感器来完成您的要求。
检索加速计传感器
Sensor accelerometer = mSensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);
只需在传感器更改事件调用时比较和复制值
@Override
public void onSensorChanged(SensorEvent event) {
if (event.sensor.getType() == Sensor.TYPE_ACCELEROMETER)
mGravity = event.values;
}
然后您可以使用以下功能获取所有轴的传感器值。
public int[] getDeviceAngles() {
float[] g = mGravity.clone();
double normOfG = Math.sqrt(g[0] * g[0] + g[1] * g[1] + g[2] * g[2]);
// Normalize the accelerometer vector
g[0] = (float) (g[0] / normOfG);
g[1] = (float) (g[1] / normOfG);
g[2] = (float) (g[2] / normOfG);
int x = (int) Math.round(Math.toDegrees(Math.atan2(g[1], g[0])));
int pitch = (int) Math.round(Math.toDegrees(Math.atan2(g[1], g[2])));
int rollValue = (int) Math.round(Math.toDegrees(Math.atan2(g[2], g[0])));
int pitchValue = pitch * -1;
int[] values = new int[3];
values[0] = x;
values[1] = pitchValue;
values[2] = rollValue;
//values contains: azimut, pitch and roll
return values;
}