วิธีตั้งค่าการวางแนวกล้อง Android ให้ถูกต้อง?


90

ฉันต้องการตั้งค่าการวางแนวกล้องตามการวางแนวอุปกรณ์ใน Android แต่ดูเหมือนจะไม่มีอะไรทำงาน ฉันพยายามหมุน Surface เช่นเดียวกับพารามิเตอร์ของกล้อง แต่การแสดงตัวอย่างกล้องในโหมดแนวตั้งจะกลับหัวเสมอ ฉันจะต้องหมุนตามเข็มนาฬิกา 90 องศาเพื่อให้ถูกต้อง นี่คือรหัสที่ฉันใช้ตอนนี้ซึ่งใช้งานได้ในโหมดแนวนอนเท่านั้น

    SurfaceHolder.Callback surfaceCallback = new SurfaceHolder.Callback() {

    @Override
    public void surfaceDestroyed(SurfaceHolder holder) {
        camera.stopPreview();
        camera.release();
        camera = null;
    }

    @Override
    public void surfaceCreated(SurfaceHolder holder) {          
        initCamera();           
    }

    private Size getOptimalPreviewSize(List<Size> sizes, int w, int h) {
        final double ASPECT_TOLERANCE = 0.2;
        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) {
            Log.d(TAG, "Checking size " + size.width + "w " + size.height
                    + "h");
            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;
    }

    @Override
    public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
        Camera.Parameters parameters = camera.getParameters();

        List<Size> sizes = parameters.getSupportedPreviewSizes();
        Size optimalSize = getOptimalPreviewSize(sizes, width, height);         
        Log.d(TAG, "Surface size is " + width + "w " + height + "h");
        Log.d(TAG, "Optimal size is " + optimalSize.width + "w " + optimalSize.height + "h");           
        parameters.setPreviewSize(optimalSize.width, optimalSize.height);           
        // parameters.setPreviewSize(width, height);            
        camera.setParameters(parameters);
        camera.startPreview();
    }
};  

6
AFAIK การแสดงตัวอย่างกล้องใช้งานได้จริงในแนวนอนเท่านั้นอย่างน้อยสำหรับ 2.2 ขึ้นไป ฉันคาดเดาว่านั่นเป็นสาเหตุที่กิจกรรมที่แสดงตัวอย่างกล้องมักจะซ่อนแถบการแจ้งเตือนของระบบและไม่มีชื่อเรื่อง ... แม้ว่าจะเป็นภาพบุคคล แต่ฉันก็คิดว่าเป็นแนวนอน "จริงๆ"
Reuben Scratton

คำตอบ:


71

จากสมาชิกคนอื่นและปัญหาของฉัน:

ปัญหาการหมุนกล้องขึ้นอยู่กับอุปกรณ์ที่แตกต่างกันและบางเวอร์ชัน

เวอร์ชัน 1.6: เพื่อแก้ไขปัญหาการหมุนและเป็นสิ่งที่ดีสำหรับอุปกรณ์ส่วนใหญ่

if (getResources().getConfiguration().orientation == Configuration.ORIENTATION_PORTRAIT)
        {   
            p.set("orientation", "portrait");
            p.set("rotation",90);
        }
        if (getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE)
        {                               
            p.set("orientation", "landscape");          
            p.set("rotation", 90);
        }

เวอร์ชัน 2.1: ขึ้นอยู่กับประเภทของอุปกรณ์ตัวอย่างเช่นไม่สามารถแก้ไขปัญหากับ XPeria X10 ได้ แต่เหมาะสำหรับ X8 และ Mini

Camera.Parameters parameters = camera.getParameters();
parameters.set("orientation", "portrait");
camera.setParameters(parameters);

เวอร์ชัน 2.2: ไม่ใช่สำหรับอุปกรณ์ทั้งหมด

camera.setDisplayOrientation(90);

http://code.google.com/p/android/issues/detail?id=1193#c42


3
คุณหมายถึงอะไรในเวอร์ชันที่นี่?
Md. Sulayman

31

จาก Javadocs สำหรับsetDisplayOrientation(int)(ต้องการ API ระดับ 9):

 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);
 }

1
@Derzu คลาสCameraInfoไม่ได้รับการแนะนำจนถึงระดับ API 9 ดังนั้นวิธีที่ฉันโพสต์ต้องใช้ API ระดับ 9
Jason Robinson

2
result = (360 - result) % 360; // compensate the mirrorควรลบภาพจากกล้องหน้าจะหมุนไม่ถูกต้อง
stevo.mit

@ stevo.mit คุณได้ยืนยันสิ่งนี้บนอุปกรณ์หลายเครื่องหรือไม่ ฉันใช้รหัสนี้ในการหมุนหลายครั้งและไม่พบการหมุนที่ไม่ถูกต้อง
Jason Robinson

2
@ Jason Robinson ฉันมีรายการโมเดลที่ระดับ api สูงกว่า 9 แต่วิธีนี้ก็ยังไม่มีผล ไม่ทราบสภาพปัญหาที่เกี่ยวข้องกับฮาร์ดแวร์ รายการอุปกรณ์ rotation_issue_models = Arrays.asList ("GT-S5360", "GT-S6802", "GT-S5830C", "GT-S5830I", "DROID2", "GLOBAL", "XT557", "Desire HD", " PC36100 "," GT-I9000 "," ADR6350 "," Mi-One Plus "," SGH-T989 "," GT-I9100 "," GT-I9001 ");
Vikram

1
@AbdulMohsin ดูที่developer.android.com/reference/android/hardware/…โดยเฉพาะ CAMERA_FACING_BACK และ CAMERA_FACING_FRONT
Jason Robinson

25

โซลูชันนี้จะใช้ได้กับAndroid ทุกเวอร์ชัน คุณสามารถใช้การสะท้อนใน Java เพื่อให้ใช้งานได้กับอุปกรณ์ Android ทั้งหมด:

โดยทั่วไปคุณควรสร้างผ้าคลุมภาพสะท้อนเพื่อเรียกใช้ Android 2.2 setDisplayOrientation แทนที่จะเรียกวิธีการเฉพาะ

วิธีการ:

    protected void setDisplayOrientation(Camera camera, int angle){
    Method downPolymorphic;
    try
    {
        downPolymorphic = camera.getClass().getMethod("setDisplayOrientation", new Class[] { int.class });
        if (downPolymorphic != null)
            downPolymorphic.invoke(camera, new Object[] { angle });
    }
    catch (Exception e1)
    {
    }
}

และแทนที่จะใช้camera.setDisplayOrientation (x) ให้ใช้setDisplayOrientation (camera, x) :

    if (Integer.parseInt(Build.VERSION.SDK) >= 8)
        setDisplayOrientation(mCamera, 90);
    else
    {
        if (getResources().getConfiguration().orientation == Configuration.ORIENTATION_PORTRAIT)
        {
            p.set("orientation", "portrait");
            p.set("rotation", 90);
        }
        if (getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE)
        {
            p.set("orientation", "landscape");
            p.set("rotation", 90);
        }
    }   

1
ส่วนอื่นใช้งานได้กับอุปกรณ์ 2.1 บางรุ่น แต่ไม่ใช่ทั้งหมด (ดูคำอธิบายของ maydenec ด้านบน)
Eric Chen

1
ผมคิดว่าหน้าเป็น Camera.Parameterstypre ลองเพิ่มบรรทัดต่อไปนี้:Camera.Parameters p = camera.getParameters();
Sehrish Khan

6

ฉันประสบปัญหาเมื่อใช้ ZBar เพื่อสแกนในแท็บ ปัญหาการวางแนวกล้อง การใช้รหัสด้านล่างฉันสามารถแก้ไขปัญหาได้ นี่ไม่ใช่ข้อมูลโค้ดทั้งหมดโปรดรับความช่วยเหลือจากสิ่งนี้เท่านั้น

 public void surfaceChanged(SurfaceHolder holder, int format, int width,
                               int height) {
     if (isPreviewRunning) {
            mCamera.stopPreview();
        }

 setCameraDisplayOrientation(mCamera);

        previewCamera();

    }



 public void previewCamera() {

        try {
            // Hard code camera surface rotation 90 degs to match Activity view
            // in portrait
            mCamera.setPreviewDisplay(mHolder);
            mCamera.setPreviewCallback(previewCallback);
            mCamera.startPreview();
            mCamera.autoFocus(autoFocusCallback);
            isPreviewRunning = true;
        } catch (Exception e) {
            Log.d("DBG", "Error starting camera preview: " + e.getMessage());
        }


    }


public void setCameraDisplayOrientation(android.hardware.Camera camera) {
        Camera.Parameters parameters = camera.getParameters();

        android.hardware.Camera.CameraInfo camInfo =
                new android.hardware.Camera.CameraInfo();
        android.hardware.Camera.getCameraInfo(getBackFacingCameraId(), camInfo);


        Display display = ((WindowManager) context.getSystemService(Context.WINDOW_SERVICE)).getDefaultDisplay();
        int rotation = display.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 (camInfo.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {
            result = (camInfo.orientation + degrees) % 360;
            result = (360 - result) % 360;  // compensate the mirror
        } else {  // back-facing
            result = (camInfo.orientation - degrees + 360) % 360;
        }
        camera.setDisplayOrientation(result);
    }




    private int getBackFacingCameraId() {
        int cameraId = -1;
        // Search for the front facing camera
        int numberOfCameras = Camera.getNumberOfCameras();
        for (int i = 0; i < numberOfCameras; i++) {
            Camera.CameraInfo info = new Camera.CameraInfo();
            Camera.getCameraInfo(i, info);
            if (info.facing == Camera.CameraInfo.CAMERA_FACING_BACK) {

                cameraId = i;
                break;
            }
        }
        return cameraId;
    }

5

ในที่สุดฉันก็แก้ไขสิ่งนี้โดยใช้แอพกล้องของ Google ได้รับการวางแนวของโทรศัพท์โดยใช้เซ็นเซอร์แล้วตั้งค่าแท็ก EXIF ​​อย่างเหมาะสม JPEG ที่ออกมาจากกล้องจะไม่ถูกปรับโฟกัสโดยอัตโนมัติ

นอกจากนี้ตัวอย่างกล้องยังทำงานได้อย่างถูกต้องในโหมดแนวนอนเท่านั้น หากคุณต้องการให้เค้าโครงกิจกรรมของคุณเป็นแนวตั้งคุณจะต้องดำเนินการด้วยตนเองโดยใช้ค่าจากเซ็นเซอร์การวางแนว


2
เฮ้วิธีรับการวางแนวกล้องโดยใช้เซ็นเซอร์ปรับทิศทาง ?? กรุณาแบ่งปันรหัสของคุณ ...
Rishi

@Rishi ดูลิงค์นี้stackoverflow.com/questions/9055460/…
PiyushMishra

4
ขอบคุณสำหรับความช่วยเหลือ แต่ปัญหาของฉันคือเมื่อฉันถ่ายภาพบนโทรศัพท์ symsung ในโหมดแนวตั้งในขณะนั้นภาพถ่ายจะแสดงหมุน 90 องศาบนหน้าจอของฉัน ดังนั้นฉันพยายามวางแนวของกล้องดังนั้นฉันจึงหมุนภาพถ่าย 90 องศาสำหรับโหมดแนวตั้ง
Rishi

5

ลองดูวิธีนี้

 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);
    }

ขอขอบคุณที่จัดการกล้องหน้า สิ่งนี้ช่วยแก้ปัญหาของฉันได้
Louis CAD

ยินดีต้อนรับคุณ @LouisCAD ยินดีที่เห็นว่าโซลูชันของฉันสามารถแก้ปัญหาของคุณได้
Mudassir Khan

4

ปัญหานี้ได้รับการแก้ไขเมื่อนานมาแล้ว แต่ฉันพบปัญหาในการรวบรวมทุกชิ้นเข้าด้วยกันดังนั้นนี่คือทางออกสุดท้ายของฉันฉันหวังว่านี่จะช่วยคนอื่นได้:

public void startPreview() {
        try {
            Log.i(TAG, "starting preview: " + started);

            // ....
            Camera.CameraInfo camInfo = new Camera.CameraInfo();
            Camera.getCameraInfo(cameraIndex, camInfo);
            int cameraRotationOffset = camInfo.orientation;
            // ...

            Camera.Parameters parameters = camera.getParameters();
            List<Camera.Size> previewSizes = parameters.getSupportedPreviewSizes();
            Camera.Size previewSize = null;
            float closestRatio = Float.MAX_VALUE;

            int targetPreviewWidth = isLandscape() ? getWidth() : getHeight();
            int targetPreviewHeight = isLandscape() ? getHeight() : getWidth();
            float targetRatio = targetPreviewWidth / (float) targetPreviewHeight;

            Log.v(TAG, "target size: " + targetPreviewWidth + " / " + targetPreviewHeight + " ratio:" + targetRatio);
            for (Camera.Size candidateSize : previewSizes) {
                float whRatio = candidateSize.width / (float) candidateSize.height;
                if (previewSize == null || Math.abs(targetRatio - whRatio) < Math.abs(targetRatio - closestRatio)) {
                    closestRatio = whRatio;
                    previewSize = candidateSize;
                }
            }

            int rotation = getWindowManager().getDefaultDisplay().getRotation();
            int degrees = 0;
            switch (rotation) {
            case Surface.ROTATION_0:
                degrees = 0;
                break; // Natural orientation
            case Surface.ROTATION_90:
                degrees = 90;
                break; // Landscape left
            case Surface.ROTATION_180:
                degrees = 180;
                break;// Upside down
            case Surface.ROTATION_270:
                degrees = 270;
                break;// Landscape right
            }
            int displayRotation;
            if (isFrontFacingCam) {
                displayRotation = (cameraRotationOffset + degrees) % 360;
                displayRotation = (360 - displayRotation) % 360; // compensate
                                                                    // the
                                                                    // mirror
            } else { // back-facing
                displayRotation = (cameraRotationOffset - degrees + 360) % 360;
            }

            Log.v(TAG, "rotation cam / phone = displayRotation: " + cameraRotationOffset + " / " + degrees + " = "
                    + displayRotation);

            this.camera.setDisplayOrientation(displayRotation);

            int rotate;
            if (isFrontFacingCam) {
                rotate = (360 + cameraRotationOffset + degrees) % 360;
            } else {
                rotate = (360 + cameraRotationOffset - degrees) % 360;
            }

            Log.v(TAG, "screenshot rotation: " + cameraRotationOffset + " / " + degrees + " = " + rotate);

            Log.v(TAG, "preview size: " + previewSize.width + " / " + previewSize.height);
            parameters.setPreviewSize(previewSize.width, previewSize.height);
            parameters.setRotation(rotate);
            camera.setParameters(parameters);
            camera.setPreviewDisplay(mHolder);
            camera.startPreview();

            Log.d(TAG, "preview started");

            started = true;
        } catch (IOException e) {
            Log.d(TAG, "Error setting camera preview: " + e.getMessage());
        }
    }

'ฉันหวังว่านี่จะช่วยคนอื่นได้' ไม่มันจะไม่ โค้ดส่วนหนึ่งของคุณถูกดึงออกจากบริบท ตัวอย่างเช่น 'isFrontFacingCam' มาจากไหน?
seanpj

4
ฉันคิดว่านี่เป็นอีกเรื่องหนึ่งจริงๆแล้วนี่ไม่มีอะไรเทียบกับปัญหาเริ่มต้นเลย คุณสามารถดาวน์โหลดได้จาก android.hardware.Camera.CameraInfo.facing == CameraInfo.CAMERA_FACING_FRONT ขออภัยเกี่ยวกับเรื่องนี้
Louis GRIGNON
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.