将相机SurfaceView旋转为竖屏

14
我查了一些有关使用SurfaceView更改相机方向的帖子,但是我的代码来自以下示例:

http://developer.android.com/resources/samples/ApiDemos/src/com/example/android/apis/graphics/CameraPreview.html

提供尺寸的函数看起来像这样...
    private Size getOptimalPreviewSize(List<Size> sizes, int w, int h) {
        final double ASPECT_TOLERANCE = 0.1;
        double targetRatio = (double) w / h;
        if (sizes == null) return null;

        Size optimalSize = null;
        double minDiff = Double.MAX_VALUE;

        int targetHeight = h;

        // Try to find an size match aspect ratio and size
        for (Size size : sizes) {
            double ratio = (double) size.width / size.height;
            if (Math.abs(ratio - targetRatio) > ASPECT_TOLERANCE) continue;
            if (Math.abs(size.height - targetHeight) < minDiff) {
                optimalSize = size;
                minDiff = Math.abs(size.height - targetHeight);
            }
        }

        // Cannot find the one match the aspect ratio, ignore the requirement
        if (optimalSize == null) {
            minDiff = Double.MAX_VALUE;
            for (Size size : sizes) {
                if (Math.abs(size.height - targetHeight) < minDiff) {
                    optimalSize = size;
                    minDiff = Math.abs(size.height - targetHeight);
                }
            }
        }
        return optimalSize;
    }

我遇到的问题是当我改变设备方向时,预览图片仍然横屏显示。 我尝试设置相机的方向,但结果非常奇怪。 有谁知道我需要更改什么以使其正确旋转?

7个回答

33

调整相机预览方向的代码有点复杂,因为它必须考虑以下因素:

  1. 传感器和设备的“自然”方向的相对方向(手机为纵向,平板电脑为横向,通常如此)
  2. 设备当前的 UI 方向(纵向、横向或每个方向的反向)
  3. 所讨论的摄像头是前置还是后置摄像头(由于前置摄像头的预览流水平镜像)

Camera.setDisplayOrientation 的文档有关于如何正确处理这些问题的示例代码。我在这里复制一遍:

public static void setCameraDisplayOrientation(Activity activity,
     int cameraId, android.hardware.Camera camera) {

   android.hardware.Camera.CameraInfo info = 
       new android.hardware.Camera.CameraInfo();

   android.hardware.Camera.getCameraInfo(cameraId, info);

   int rotation = activity.getWindowManager().getDefaultDisplay().getRotation();
   int degrees = 0;

   switch (rotation) {
       case Surface.ROTATION_0: degrees = 0; break;
       case Surface.ROTATION_90: degrees = 90; break;
       case Surface.ROTATION_180: degrees = 180; break;
       case Surface.ROTATION_270: degrees = 270; break;
   }

   int result;
   if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {
       result = (info.orientation + degrees) % 360;
       result = (360 - result) % 360;  // compensate the mirror
   } else {  // back-facing
       result = (info.orientation - degrees + 360) % 360;
   }
   camera.setDisplayOrientation(result);
}

在UI绘制完成后调用此方法(可以使用onSurfaceChanged作为指示器),或者当设备UI进行旋转时调用此方法(可以使用onConfigurationChanged作为指示器)。


如何在onSurfaceChanged中设置这个方法? - Vishal Senjaliya
请解释一下方法参数中的相机ID是什么意思。 - Najaf Ali
cameraId = 0 表示后置摄像头,cameraId = 1 表示前置摄像头。但是这种方法并不适用于所有设备。 - h8pathak

9
我可以通过在onSurfaceChanged()中插入以下代码来解决旋转问题:
    if (mHolder.getSurface() == null) {
        // preview surface does not exist
        return;
    }

    // stop preview before making changes
    try {
        mCamera.stopPreview();
    } catch (Exception e) {
        // ignore: tried to stop a non-existent preview
    }

    // make any resize, rotate or reformatting changes here
    if (this.getResources().getConfiguration().orientation != Configuration.ORIENTATION_LANDSCAPE) {

        mCamera.setDisplayOrientation(90);

    } else {

        mCamera.setDisplayOrientation(0);

    }
    // start preview with new settings
    try {
        mCamera.setPreviewDisplay(mHolder);
        mCamera.startPreview();

    } catch (Exception e) {
        Log.d(TAG, "Error starting camera preview: " + e.getMessage());
    }

但是这又带来了另一个问题,或者更好地说,没有解决一个问题 - 尽管方向正确,预览图像仍然只占用与横向视图相同的空间。最终我放弃了,并通过以下方式强制使用横向模式:

setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);

在onCreate()方法中,我为横屏设计了我的布局。祝你好运,希望有人能解决这个次要问题!


7

你可以试试这个,但是我在三星Galaxy Tab上试过了。

public void surfaceCreated(SurfaceHolder holder)
{
   // The Surface has been created, acquire the camera and tell it where to draw.
   mCamera = Camera.open();

   Parameters params = mCamera.getParameters();

   if (this.getResources().getConfiguration().orientation != Configuration.ORIENTATION_LANDSCAPE)
   {
    params.set("orientation", "portrait");
    mCamera.setDisplayOrientation(90);
   }

    try
      {
      mCamera.setPreviewDisplay(holder);
      }
      catch (IOException exception)
      {
        mCamera.release();
        mCamera = null;
      }

}

我目前没有工作环境来测试这个(我玩弄它太多了),但是一旦我把它建立起来,我会尝试这个。 - frak
params.set("orientation", "portrait"); mCamera.setDisplayOrientation(90); 方向就像宝石一样好用! - vss
谢谢@VIGNESH.SRINIVASAGAN,你救了我的一天,伙计!!! - Ravi Vaniya

2
请尝试这样做...
@Override
 public void surfaceChanged(SurfaceHolder holder,
                       int format, int width, int height)
{
// TODO Auto-generated method stub

if(previewing)
{
    camera.stopPreview();
    previewing = false;
}
 Camera.Parameters parameters = camera.getParameters();
      Display display = ((WindowManager)getSystemService(WINDOW_SERVICE)).getDefaultDisplay();
   int or=cameraInfo.orientation;
     // You need to choose the most appropriate previewSize for your app
   // .... select one of previewSizes here
           /* parameters.setPreviewSize(previewSize.width, previewSize.height);*/
   if(display.getRotation() == Surface.ROTATION_0)
    {

    camera.setDisplayOrientation(90);
    or=90;
    }

    if(display.getRotation() == Surface.ROTATION_180)
    {
        camera.setDisplayOrientation(270);
    or=270;
    }
    if(display.getRotation() == Surface.ROTATION_270)
    {
        camera.setDisplayOrientation(180);
        or=180;
    }

  parameters.setRotation(or);

  camera.setParameters(parameters);
 try
{
        camera.setPreviewDisplay(cameraSurfaceHolder);
        camera.startPreview();
        previewing = true;
    }
    catch (IOException e)
    {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}

2

好的,我来做!

我已经在surfaceCreated()方法中以以下方式解决了这个问题:

 public void surfaceCreated(SurfaceHolder holder) {
    try {

        camera = Camera.open();
    } catch (RuntimeException e) {
        System.err.println(e);
        return;
    }
    Camera.Parameters param;
    param = camera.getParameters();
    if (this.getResources().getConfiguration().orientation != Configuration.ORIENTATION_LANDSCAPE)
    {
        param.set("orientation", "portrait");
        setCameraDisplayOrientation(this,1,camera);
    }
    camera.setParameters(param);

    try {
        camera.setPreviewDisplay(surfaceHolder);
        camera.startPreview();
    } catch (Exception e) {
        System.err.println(e);
        return;
    }
}

在下面添加此方法

 public static void setCameraDisplayOrientation(Activity activity,
                                               int cameraId, android.hardware.Camera camera) {

    android.hardware.Camera.CameraInfo info =
            new android.hardware.Camera.CameraInfo();

    android.hardware.Camera.getCameraInfo(cameraId, info);

    int rotation = activity.getWindowManager().getDefaultDisplay().getRotation();
    int degrees = 0;

    switch (rotation) {
        case Surface.ROTATION_0: degrees = 0; break;
        case Surface.ROTATION_90: degrees = 90; break;
        case Surface.ROTATION_180: degrees = 180; break;
        case Surface.ROTATION_270: degrees = 270; break;
    }

    int result;
    if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {
        result = (info.orientation + degrees) % 360;
        result = (360 - result) % 360;  // compensate the mirror
    } else {  // back-facing
        result = (info.orientation - degrees + 360) % 360;
    }
    camera.setDisplayOrientation(result);
}

3:在调用此活动之前,请确保您已经在一个活动中添加了相机权限以支持Nougat。不要在此活动中调用相机权限,因为SurfaceCreated在活动创建时被调用,您的权限将挂起打开相机。


嗨,最近的朋友,你来访问这个答案了吗?我能帮你更多吗?请在这里输入信息。 - Najaf Ali

2
我通过添加以下内容解决了这个问题:
setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_PORTRAIT);

到 onCreate 事件。


-1

只需添加这个

mCamera.setDisplayOrientation(90);

这会添加任何新内容吗? - Laurenz Albe

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接