5

メソッド surfaceChanged() から Camera.setParameters() を呼び出すと、RuntimeException エラーが発生します。同じ問題に関する他の投稿を見ましたが、明確な解決策が見つかりませんでした: パラメータをカメラに設定する際の問題

私の質問は次のとおりです。

  1. surfaceCreated() の直後に surfaceChanged() を呼び出す必要があるのはなぜですか? surfaceCreated() でプレビューを開始し、それを停止して surfaceChanged() で別のプレビューを再開するのは (少なくとも私にとっては) 直観に反します。

  2. 初めて surfaceChanged() が呼び出されたとき、渡される幅と高さは何によって決定されますか? 前述の投稿で示唆され、私の実験で検証されたように、値は必ずしも Camera.setParameters() でサポートされているわけではありません。私のコード スニペットとログ情報については、以下を参照してください。

  3. surfaceChanged() が呼び出される前に、幅と高さの値が Camera.setParameters() によってサポートされることを保証する方法はありますか? それを確実にするために、surfaceChanged() 内にコードを追加することができます。たとえば、w と h は、サポートされているすべてのプレビュー サイズと比較して、サポートされている最も近い値に変更できます。ただし、これを行うとパフォーマンスが低下します。

以下のリストは、私のコード スニペットと LogCat 出力です。ログ情報は、surfaceChanged() 内の Camera.setParameters() がエラーの原因であることを示しています。setParameters() が呼び出された行をコメントアウトすると、エラー メッセージが表示されなくなります。

[編集: Amazon Kindle Fire HD でテストしています。Android のバージョンは 4.0.3、API 15 です]

コードスニペット:

public void surfaceCreated(SurfaceHolder holder) {
    // Log & debug
    Log.d(TAG, "surfaceCreated() is called");

    // The Surface has been created, now tell the camera where to draw the preview.
    try {
        mCamera.setPreviewDisplay(mHolder);
        mCamera.startPreview();
    } catch (IOException e) {
        Log.d(TAG, "surfaceCreated(): " + e.getMessage());
    }

    if (mHolder.getSurface() == null) {
        Log.d(TAG, "surfaceCreated(): preview surface does not exist yet!");
    }
}

public void surfaceDestroyed(SurfaceHolder holder) {
    // Log & debug
    Log.d(TAG, "surfaceDestroyed() is called");

    // Stop preview and release camera 
    if (holder.getSurface() != null) {
        mCamera.stopPreview();
        //mCamera.release();  //should we release camera here?
        //mCamera = null;
    }
    else {
        Log.d(TAG, "surfaceDestroyed(): preview surface does not exist");
    }
}

public void surfaceChanged(SurfaceHolder holder, int format, int w, int h) {
    // Log and debug
    Log.d(TAG, "surfaceChanged() is called with width = " + w);
    Log.d(TAG, "surfaceChanged() is called with height = " + h);

    // If your preview can change or rotate, take care of those events here.
    // Make sure to stop the preview before resizing or reformatting it.
    if (holder.getSurface() == null) {
      Log.d(TAG, "surfaceChanged(): preview surface does not exist");
      return;
    }

    // stop preview before making changes
    try {
        mCamera.stopPreview();
    } catch (Exception e) {
        Log.d(TAG, "surfaceChanged(): tried to stop a non-existent preview " + e.getMessage());
    }

    // set preview size and make any resize, rotate or
    // reformatting changes here
    Camera.Parameters params = mCamera.getParameters();
    List<Size> mSizes = params.getSupportedPreviewSizes();

    // list all 25 supported preview sizes
    // i==25 will cause an "IndexOutofBoundsException"
    for (int i=0; i<25; i++) {
        Size mSize  = mSizes.get(i);
        if (mSize == null) break;
        Log.d(TAG, "surfaceChanged() i = " + i);
        Log.d(TAG, "surfaceChanged() preview width = " + mSize.width);
        Log.d(TAG, "surfaceChanged() preview height = " + mSize.height);
    }
    Size       mSize  = mSizes.get(0);
    if(mSize != null) { //make sure we don't pull a NullException.
        if (w > mSize.width) w  = mSize.width;
        if (h > mSize.height) h = mSize.height;
        params.setPreviewSize(w, h); //set the size, since we know it. 
    } 
    mCamera.setParameters(params); //set the parameters now. 

    // start preview with new settings
    try {
        mCamera.setPreviewDisplay(holder);
        mCamera.startPreview();
    } catch (Exception e){
        Log.d(TAG, "surfaceChanged() restart preview " + e.getMessage());
    }
}

LogCat 出力:

02-13 10:46:08.113: D/CameraPreview(2886): surfaceCreated() is called
02-13 10:46:08.496: D/CameraPreview(2886): surfaceChanged() is called with width = 1072
02-13 10:46:08.496: D/CameraPreview(2886): surfaceChanged() is called with height = 705
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 0
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 1280
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 720
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 1
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 960
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 720
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 2
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 800
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 480
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 3
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 720
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 576
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 4
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 720
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 480
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 5
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 768
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 576
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 6
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 640
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 480
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 7
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 320
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 240
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 8
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 352
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 288
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 9
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 240
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 160
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 10
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 176
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 144
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 11
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 160
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 120
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 12
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 128
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 96
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 13
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 720
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 1280
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 14
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 480
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 800
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 15
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 576
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 720
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 16
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 576
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 768
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 17
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 480
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 720
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 18
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 480
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 640
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 19
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 288
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 352
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 20
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 240
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 320
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 21
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 240
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 160
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 22
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 144
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 176
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 23
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 120
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 160
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() i = 24
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview width = 96
02-13 10:46:08.941: D/CameraPreview(2886): surfaceChanged() preview height = 128
02-13 10:46:08.957: D/AndroidRuntime(2886): Shutting down VM
02-13 10:46:08.957: W/dalvikvm(2886): threadid=1: thread exiting with uncaught exception (group=0x40a6c1f8)
02-13 10:46:08.965: E/AndroidRuntime(2886): FATAL EXCEPTION: main
02-13 10:46:08.965: E/AndroidRuntime(2886): java.lang.RuntimeException: setParameters failed
02-13 10:46:08.965: E/AndroidRuntime(2886):     at android.hardware.Camera.native_setParameters(Native Method)
02-13 10:46:08.965: E/AndroidRuntime(2886):     at android.hardware.Camera.setParameters(Camera.java:1295)
02-13 10:46:08.965: E/AndroidRuntime(2886):     at com.xyz.camera.CameraPreview.surfaceChanged(CameraPreview.java:109)
02-13 10:46:08.965: E/AndroidRuntime(2886):     at android.view.SurfaceView.updateWindow(SurfaceView.java:544)
02-13 10:46:08.965: E/AndroidRuntime(2886):     at android.view.SurfaceView.access$000(SurfaceView.java:81)
02-13 10:46:08.965: E/AndroidRuntime(2886):     at android.view.SurfaceView$3.onPreDraw(SurfaceView.java:169)
02-13 10:46:08.965: E/AndroidRuntime(2886):     at android.view.ViewTreeObserver.dispatchOnPreDraw(ViewTreeObserver.java:590)
02-13 10:46:08.965: E/AndroidRuntime(2886):     at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:1630)
02-13 10:46:08.965: E/AndroidRuntime(2886):     at android.view.ViewRootImpl.handleMessage(ViewRootImpl.java:2462)
02-13 10:46:08.965: E/AndroidRuntime(2886):     at android.os.Handler.dispatchMessage(Handler.java:99)
02-13 10:46:08.965: E/AndroidRuntime(2886):     at android.os.Looper.loop(Looper.java:137)
02-13 10:46:08.965: E/AndroidRuntime(2886):     at android.app.ActivityThread.main(ActivityThread.java:4486)
02-13 10:46:08.965: E/AndroidRuntime(2886):     at java.lang.reflect.Method.invokeNative(Native Method)
02-13 10:46:08.965: E/AndroidRuntime(2886):     at java.lang.reflect.Method.invoke(Method.java:511)
02-13 10:46:08.965: E/AndroidRuntime(2886):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:784)
02-13 10:46:08.965: E/AndroidRuntime(2886):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:551)
02-13 10:46:08.965: E/AndroidRuntime(2886):     at dalvik.system.NativeStart.main(Native Method)
4

1 に答える 1

0

これを取得する可能性のある問題の 1 つは、API 15 を使用しているため、API 11 からの getSupportedVideoSizes() を使用する必要があるという事実です。特定のデバイス (実行中のデバイスハニカム以上) プレビューのサイズとサポートされているビデオのサイズが常に同じであるとは限らないためです。

編集::あなたのコメントに関して:テストデバイスでサポートされているすべてのプレビューサイズを出力します。あなたがすでに持っているもの。次に、1つ(例:取得した秒)を取得し、テスト目的で静的に設定します

例えば:

for int (i = 0; i < mSizes.size(); i ++)
    {
        Log each size;
    }

そこからランダムなサイズを取得します:例: 480 x 800 と設定します

    params.setPreviewSize(480, 800);

これにより、プレビュー サイズを設定すると適切なサイズが表示されるかどうかが確実にわかります。

EDIT2:コメントを明確にするために、回答を再度更新します

onSurfaceChanged 内には 2 つの変数があります

int width = 0;
int height = 0;

次に、ifステートメントで

if (w > mSize.width) width = mSize.width;
if (h > mSize.height) height = mSize.height;

また、あなたがしていることは、

mSizes.get(mSize.size() -1)。w と h は常に mSize.width と mSize.height より大きくなるからです

EDIT 3:あなたのコメントに基づいてOK. onSufaceCreated() 内でカメラを初期化し、setPreviewSize(w,h) を setPreviewSize(mSizes.get(mSizes.size() -1).width, mSizes.get(mSizes.size() -1) に変更します。身長);

次に、任意のサイズに surfaceView を設定する関数を追加します ()

surfaceParams = surfaceView.getLayoutParameters();
surfaceParams.setWidth(somewidthyouwant);
surfaceParams.setHeight(someHeightYouWant);
surfaceView.setLayoutParams(surfaceParams);

これにより、サーフェスビューが必要なサイズに設定されます。

于 2013-02-13T19:53:26.550 に答える