Android: ориентация предварительного просмотра камеры в HTC EVO (Android 2.1 или 2.2)

Я разрабатываю приложение для Android, которое использует API камеры, используя HTC EVO в качестве тестового устройства. Независимо от того, что я пробовал до сих пор, единственный раз, когда предварительный просмотр камеры выглядит правильно, — это в ландшафтном режиме (точнее, с поворотом на 90 градусов). Кажется, что нет возможности правильно сориентировать предварительный просмотр в портретном режиме (поворот на 0 градусов).

Приложение камеры по умолчанию на устройстве (для HTC Sense) без проблем допускает любое вращение, поэтому я знаю, что аппаратных ограничений нет. Я даже загрузил некоторый исходный код с сайта разработчика HTC, но, по-видимому, все это на языке C - ядро.

Может ли кто-нибудь указать мне правильное направление для решения этой проблемы? Есть ли способ правильно повернуть предварительный просмотр в Android 2.1 или 2.2?

Спасибо.

P.S. Вот код, который я использую, если это поможет...

package spikes.cameraSpike03;

import java.util.List;

import android.app.Activity;
import android.graphics.PixelFormat;
import android.hardware.Camera;
import android.hardware.Camera.Size;
import android.os.Bundle;
import android.util.Log;
import android.view.Surface;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.Window;
import android.view.WindowManager;

public class MainActivity extends Activity implements SurfaceHolder.Callback {
 private static final String LOG_TAG = "spikes.cameraSpike03 - MainActivity";

 private Camera _camera;
 private boolean _previewIsRunning = false;

 private SurfaceView _svCameraView;
 private SurfaceHolder _surfaceHolder;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);

        getWindow().setFormat(PixelFormat.TRANSLUCENT);
        getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);

        setContentView(R.layout.main);

        _svCameraView = (SurfaceView)findViewById(R.id.svCameraView);

        _surfaceHolder = _svCameraView.getHolder();
        _surfaceHolder.addCallback(this);
        _surfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
    }

 @Override
 public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
  if(_previewIsRunning){
   _camera.stopPreview();
  }

  try{
   Camera.Parameters parameters = _camera.getParameters();

   //Get the optimal preview size so we don't get an exception when setting the parameters 
   List<Size> supportedPreviewSizes = parameters.getSupportedPreviewSizes();
   Size optimalPreviewSize = getOptimalPreviewSize(supportedPreviewSizes, width, height);
   parameters.setPreviewSize(optimalPreviewSize.width, optimalPreviewSize.height);

   _camera.setParameters(parameters);

   _camera.setPreviewDisplay(holder);
  }
  catch(Exception ex){
   ex.printStackTrace();
   Log.e(LOG_TAG, ex.toString());
  }

  _camera.startPreview();
  _previewIsRunning = true;
 }

 @Override
 public void surfaceCreated(SurfaceHolder holder) {
  _camera = Camera.open();
 }

 @Override
 public void surfaceDestroyed(SurfaceHolder holder) {
  _camera.stopPreview();
  _previewIsRunning = false;
  _camera.release();
 }

 private Size getOptimalPreviewSize(List<Size> sizes, int w, int h) {
        final double ASPECT_TOLERANCE = 0.05;
        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;
    }

}

person mahdaeng    schedule 18.01.2011    source источник


Ответы (1)


Попробуйте заблокировать ориентацию Activity по горизонтали, если это соответствует вашему дизайну, или поэкспериментируйте с этим:

private android.hardware.Camera mCamera;

...

mCamera.setDisplayOrientation(90);
params.setRotation(90);
mCamera.setParameters(params);

Но я думаю, что setDisplayOrientation может быть только 2.2...

person jsonfry    schedule 18.01.2011
comment
Спасибо, Фрай15. К сожалению, с этим есть две проблемы: 1) дизайн должен быть вертикальным (книжный) и 2) вы правы в том, что setDisplayOrientation(...) работает только на 2.2. Я посмотрю, смогу ли я убедить клиента позволить нам ориентироваться на 2.2 и выше. - person mahdaeng; 19.01.2011
comment
Я создал (портретное) приложение камеры несколько месяцев назад и провел довольно много исследований по этому поводу, до версии 2.2 не было возможности сделать это в портретном режиме. Секрет заключался в том, чтобы заблокировать активность в горизонтальном режиме, а затем подделать все, чтобы она выглядела так, как будто она была в портретной ориентации, повернув активы и т. д. Это не было для меня вариантом по разным причинам, поэтому это должно было быть только 2.2+ :/ - person jsonfry; 19.01.2011
comment
Спасибо, Фрай15. Я думаю, вы правы - это, вероятно, должно быть 2.2. - person mahdaeng; 27.01.2011
comment
@ fry15 Большое спасибо за предложение блокировки в горизонтальном режиме - простое, но эффективное. - person donturner; 05.12.2011