Android手机会影响传感器(加速度计和磁场),而手机则面向用户

时间:2016-12-21 19:21:08

标签: android android-camera android-sensors azimuth

对于我正在制作的应用程序,我需要一个摄像头和一个指南针。应用程序在清单中设置为横向模式 首先我实施了指南针。正如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度(虽然稳定)
我已经尝试过寻找解决方案(无论是我自己还是在线),但到目前为止,我的努力都是徒劳的。我想知道如何解决这个问题。
谢谢!

1 个答案:

答案 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;
}