web-dev-qa-db-ja.com

android camera2ハンドルズーム

Android Camera2APIは初めてです。すべてのプロジェクトを新しいCamera2APIに移動します。開始点として Camera2Basicの例 を使用しました。

これを追加して、ハンドルズームを試しています。

_public boolean onTouchEvent(MotionEvent event) {
    try {
        CameraManager manager = (CameraManager) getSystemService(Context.CAMERA_SERVICE);
        CameraCharacteristics characteristics = manager.getCameraCharacteristics(mCameraId);
        float maxZoom = (characteristics.get(CameraCharacteristics.SCALER_AVAILABLE_MAX_DIGITAL_ZOOM))*10;

        Rect m = characteristics.get(CameraCharacteristics.SENSOR_INFO_ACTIVE_ARRAY_SIZE);
        int action = event.getAction();
        float current_finger_spacing;

        if (event.getPointerCount() > 1) {
            // Multi touch logic
            current_finger_spacing = getFingerSpacing(event);

            if(finger_spacing != 0){
                if(current_finger_spacing > finger_spacing && maxZoom > zoom_level){
                    zoom_level++;

                }
                else if (current_finger_spacing < finger_spacing && zoom_level > 1){
                    zoom_level--;

                }
                int minW = (int) (m.width() / maxZoom);
                int minH = (int) (m.height() / maxZoom);
                int difW = m.width() - minW;
                int difH = m.height() - minH;
                int cropW = difW /100 *(int)zoom_level;
                int cropH = difH /100 *(int)zoom_level;
                cropW -= cropW & 3;
                cropH -= cropH & 3;
                Rect zoom = new Rect(cropW, cropH, m.width() - cropW, m.height() - cropH);
                mPreviewRequestBuilder.set(CaptureRequest.SCALER_CROP_REGION, zoom);
            }
            finger_spacing = current_finger_spacing;
        }
        else{
            if (action == MotionEvent.ACTION_UP) {
                //single touch logic
            }
        }

        try {
            mCaptureSession.setRepeatingRequest(mPreviewRequestBuilder.build(), mCaptureCallback,
                    null);
        }
        catch (CameraAccessException e) {
            e.printStackTrace();
        }
        catch (NullPointerException ex)
        {
            ex.printStackTrace();
        }
    }
    catch (CameraAccessException e)
    {
        throw new RuntimeException("can not access camera.", e);
    }

    return true;
}
_

この:

_private float getFingerSpacing(MotionEvent event) {
    float x = event.getX(0) - event.getX(1);
    float y = event.getY(0) - event.getY(1);
    return FloatMath.sqrt(x * x + y * y);
}
_

しかし、キャプチャした後、画像の結果はズームなしです。どうすればそれを実現できますか?皆さんありがとう。

UpdatecaptureBuilder.set(CaptureRequest.SCALER_CROP_REGION, zoom);メソッドをcaptureStillPicture()メソッドに追加する必要があります。

11
Eli Elezra

captureBuilder.set(CaptureRequest.SCALER_CROP_REGION, zoom);メソッドにcaptureStillPicture()を追加する必要があります。

6
Eli Elezra

繰り返しプレビューカメラ出力用にCaptureRequestBuilderSCALER_CROP_REGIONを設定しただけです。 CaptureRequestBuilderのJPEGImageSaverを出力として使用する同じcropregionプロパティをSurfaceに追加するだけで、すべて設定できます。

3
rcsumner

グローバル:

companion object {
    private const val ZOOM_GESTURE_SENSITIVITY = .0005f
    private const val MAX_ZOOM_FACTOR = 3f
}

private var fingerDistance: Float? = null
private var zoom = 1f

リスナー:

textureView.setOnTouchListener({ _, event ->
    if (event.action != MotionEvent.ACTION_MOVE || event.pointerCount <= 1) {
        fingerDistance = null
        return@setOnTouchListener true
    }
    val newFingerDistance = with(event) {
        val x = getX(0) - getX(1)
        val y = getY(0) - getY(1)
        sqrt(x * x + y * y) * resources.displayMetrics.density
    }
    if (fingerDistance != null) {
        val manager = getSystemService(Context.CAMERA_SERVICE) as CameraManager
        val characteristics = manager.getCameraCharacteristics(cameraId)
        val maxZoom = min(MAX_ZOOM_FACTOR, characteristics.get(
                CameraCharacteristics.SCALER_AVAILABLE_MAX_DIGITAL_ZOOM))
        val zoomChange = (newFingerDistance - fingerDistance!!) * ZOOM_GESTURE_SENSITIVITY
        zoom = min(maxZoom, max(1f, zoom + zoomChange))
        val sensorSize = characteristics.get(
                CameraCharacteristics.SENSOR_INFO_ACTIVE_ARRAY_SIZE)!!
        val cropW = (sensorSize.width() * (1 - 1 / zoom) / 2).toInt()
        val cropH = (sensorSize.width() * (1 - 1 / zoom) / 2).toInt()
        val zoomRect = Rect(cropW, cropH,
                sensorSize.width() - cropW,
                sensorSize.height() - cropH)
        previewRequestBuilder.set(CaptureRequest.SCALER_CROP_REGION, zoomRect)
    }
    fingerDistance = newFingerDistance
    try {
        captureSession
                ?.setRepeatingRequest(previewRequestBuilder.build(), captureCallback, null)
    } catch (e: Exception) {
        e.printStackTrace()
    }
    true
})
0

SCALER.CROP.REGIONcaptureBuilderに設定するだけです。あなたはこの方法でそれを行うことができます:

yourCapturebuilder.(CaptureRequest.SCALER_CROP_REGION, newZoom);

別の方法で、ズームを設定のままにしておきたい場合。次のようなことをお勧めします。Rectを文字列として設定に保存し、後でそれを回復して使用するか、カメラを開くたびに呼び出します。

Preferences.edit().putString(CameraSettings.KEY_ZOOM,newZoom.toString());

後でこのメソッドを呼び出して、ズームを動的に設定します。

public boolean setZoomValue(CaptureRequest.Builder builder) {
        Log.i(TAG,"zoom preference value " + mPreferences.getString(CameraSettings.KEY_ZOOM,null));
        String rawZoomValue =  mPreferences.getString(CameraSettings.KEY_ZOOM, null);
        if (rawZoomValue == null)
            return false;
        rawZoomValue = rawZoomValue.replaceAll("[Rect() ]", "");
        String[] rectZoomList = rawZoomValue.split(",|\\-|\\)|\\(");
        Rect zoomValue = new Rect( Integer.parseInt(rectZoomList[0]), Integer.parseInt(rectZoomList[1]), Integer.parseInt(rectZoomList[2]),Integer.parseInt(rectZoomList[3]));
        builder.set(CaptureRequest.SCALER_CROP_REGION, zoomValue);
        Log.i(TAG, "Zoom applied: " + zoomValue);
        return true;

    }