Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

CameraSourcePreview doesn't fill whole screen height #23

Open
donpironet opened this issue Sep 11, 2015 · 41 comments
Open

CameraSourcePreview doesn't fill whole screen height #23

donpironet opened this issue Sep 11, 2015 · 41 comments

Comments

@donpironet
Copy link

I'm having everything the same as in this sample except my activity layout is this:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/topLayout"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:keepScreenOn="true"
    android:weightSum="100"
    android:orientation="horizontal">

    <be.citylife.communitypurchaseapp.view.camera.CameraSourcePreview
        android:id="@+id/preview"
        android:layout_width="0dp"
        android:layout_height="match_parent"
        android:layout_weight="60">

        <be.citylife.communitypurchaseapp.view.camera.GraphicOverlay
            android:id="@+id/overlay"
            android:layout_width="match_parent"
            android:layout_height="match_parent" />

    </be.citylife.communitypurchaseapp.view.camera.CameraSourcePreview>

    <FrameLayout
        android:id="@+id/sideContainer"
        android:layout_width="0dp"
        android:layout_height="match_parent"
        android:layout_weight="40"/>

</LinearLayout>

My tablet is in landscape and I want that the cameraPreviewSource is always left and fills the whole screen in the height and then right off it I'm having a fragment that fills the rest.

This layout works except my previewsource doesn't fill the whole height (even though it shows correct in the preview). It has a black banner on it. Even my width is actually smaller than I want you can see this on the screenshot:

http://i61.tinypic.com/vctmw0.png

I played with the CameraSourcePreview with the width and height in the onLayout function but it doesn't help.

Anyone an idea how to solve this?

EDIT:

I think there is a problem in the onLayout method off the CameraSourcePreview. That it doesn't calculate it right when you use maybe layout weight or something I don't know.

@pm0733464
Copy link
Contributor

CameraSourcePreview is included in the example code. This was meant as more of a "Hello World" sample rather than a formal part of the API.

It has to account for getting the aspect ratio of the view to match the aspect ratio of the preview images from the camera, as well as accounting for the camera rotation. So it actually overrides the size of the view to end up with something reasonable. See here:

https://github.com/googlesamples/android-vision/blob/master/visionSamples/FaceTracker/app/src/main/java/com/google/android/gms/samples/vision/face/facetracker/ui/camera/CameraSourcePreview.java#L126

Feel free to copy/modify this class and change it to best meet your needs.

@donpironet
Copy link
Author

Thats what I thought. The only problem I'm facing right now is that I don't know how to get the width of the screen because my layout uses layout_weight and I don't know how to get 60 % of the width. Otherwise it would be simple I guess, just retrieve the width/height or set them. Any idea or guideline how to do this?

@lucasjj85
Copy link

I'm facing the same issue, some extra padding on landscape and portrait mode.

@abhirav
Copy link

abhirav commented Sep 14, 2015

Hi,
Comment or remove below lines from CameraSourcePreview and it should be fine. I was having same issue like you and it is solved now.

if (childHeight > layoutHeight) {
    childHeight = layoutHeight;
    childWidth = (int)(((float) layoutHeight / (float) height) * width);
}

@lucasjj85
Copy link

Thank you Abhirav,
your suggestion has completely solved the issue in landscape mode (there is still a bit of padding in portrait mode but it is ok).

@donpironet
Copy link
Author

I removed the lines to but still padding on the bottom in landscape mode. Is this some problem with layout weight?

@Kalen66100
Copy link

This is my code, no padding in portrait and fill right in landscape.

I use layoutHeight instead of childHeight in for (int i = 0; i < getChildCount(); ++i){...}

    if (mCameraSource != null)
    {
        Size size = mCameraSource.getPreviewSize();
        if (size != null)
        {
            width = size.getWidth();
            height = size.getHeight();
        }
    }

    // Swap width and height sizes when in portrait, since it will be rotated 90 degrees
    if (isPortraitMode())
    {
        int tmp = width;

        //noinspection SuspiciousNameCombination
        width = height;
        height = tmp;
    }

    final int layoutWidth = right - left;
    final int layoutHeight = bottom - top;

    // Computes height and width for potentially doing fit width.
    int childWidth = layoutWidth;
    int childHeight = (int) (((float) layoutWidth / (float) width) * height);

    for (int i = 0; i < getChildCount(); ++i)
    {
        getChildAt(i).layout(0, 0, childWidth, layoutHeight);
    }

    try
    {
        startIfReady();
    }
    catch (SecurityException se)
    {
        Log.e(TAG, "Do not have permission to start the camera", se);
    }
    catch (IOException e)
    {
        Log.e(TAG, "Could not start camera source.", e);
    }
}

@mmagician
Copy link

@abhirav Thanks, this works.

I had the same problem, unnecessary white space on the right in portrait mode.

Another thing which worked for me was removing BOTH what abhirav suggested and removing these lines:

// Computes height and width for potentially doing fit width.
int childWidth = layoutWidth;
int childHeight = (int)(((float) layoutWidth / (float) width) * height);

On the device I tested, this had no effect on how the preview actually looked, but using layoutHeight is the actual device height, whereas childHeight is computed and (in my case) bigger than device's height. But as I said, no effect on how it actually looked.

@raokarthik74
Copy link

Code suggested by mmagician and abhirav works perfectly. But, the camera preview is not in the correct ratio in portrait mode. the height is getting stretched. Could you help me out in this regard ?

wax911 pushed a commit to wax911/AeGis_Nav that referenced this issue Oct 18, 2015
-> Added Icons to Nav Drawer
-> Changed camera source Resolution, thanks to
googlesamples/android-vision#23
-> added flat icons for navigation
-> Added PolyLines for joining point on the map
-> Removed some duplicated classes
@af001
Copy link

af001 commented Oct 30, 2015

@Kalen66100
Thanks, your solution worked for me.

@carlo88doc
Copy link

Thank you @Kalen66100 , it worked for me too.

@sree973
Copy link

sree973 commented Jan 27, 2016

@Kalen66100 Thanks you. It worked for me

@romeryto
Copy link

romeryto commented Mar 7, 2016

Thank you @Kalen66100 and @abhirav . You save my code!

@pm0733464
Copy link
Contributor

The following works, while also preserving the aspect ratio. This will slightly oversize the camera display, requiring cropping along one dimension. Change CameraSourcePreview.onLayout to:

    @Override
    protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
        int previewWidth = 320;
        int previewHeight = 240;
        if (mCameraSource != null) {
            Size size = mCameraSource.getPreviewSize();
            if (size != null) {
                previewWidth = size.getWidth();
                previewHeight = size.getHeight();
            }
        }

        // Swap width and height sizes when in portrait, since it will be rotated 90 degrees
        if (isPortraitMode()) {
            int tmp = previewWidth;
            previewWidth = previewHeight;
            previewHeight = tmp;
        }

        final int viewWidth = right - left;
        final int viewHeight = bottom - top;

        int childWidth;
        int childHeight;
        int childXOffset = 0;
        int childYOffset = 0;
        float widthRatio = (float) viewWidth / (float) previewWidth;
        float heightRatio = (float) viewHeight / (float) previewHeight;

        // To fill the view with the camera preview, while also preserving the correct aspect ratio,
        // it is usually necessary to slightly oversize the child and to crop off portions along one
        // of the dimensions.  We scale up based on the dimension requiring the most correction, and
        // compute a crop offset for the other dimension.
        if (widthRatio > heightRatio) {
            childWidth = viewWidth;
            childHeight = (int) ((float) previewHeight * widthRatio);
            childYOffset = (childHeight - viewHeight) / 2;
        } else {
            childWidth = (int) ((float) previewWidth * heightRatio);
            childHeight = viewHeight;
            childXOffset = (childWidth - viewWidth) / 2;
        }

        for (int i = 0; i < getChildCount(); ++i) {
            // One dimension will be cropped.  We shift child over or up by this offset and adjust
            // the size to maintain the proper aspect ratio.
            getChildAt(i).layout(
                    -1 * childXOffset, -1 * childYOffset,
                    childWidth - childXOffset, childHeight - childYOffset);
        }

        try {
            startIfReady();
        } catch (IOException e) {
            Log.e(TAG, "Could not start camera source.", e);
        }
    }

@namphho
Copy link

namphho commented May 25, 2016

Hi all, this is my solution and It worked on SS Note 3, Moto G. I hope it is useful for you
in BarcodeCaptureActivity.java - line 204

DisplayMetrics metrics = new DisplayMetrics();
        getWindowManager().getDefaultDisplay().getMetrics(metrics);
        Log.e(TAG, "widthPixels: " + metrics.widthPixels + " -- metrics: " + metrics.heightPixels);
        CameraSource.Builder builder = new CameraSource.Builder(getApplicationContext(), barcodeDetector)
                .setFacing(CameraSource.CAMERA_FACING_BACK)
                .setRequestedPreviewSize(metrics.heightPixels, metrics.widthPixels)
                .setRequestedFps(15.0f);

in CameraSourcePreview.java - replace childWidth, childHeight by width, height

for (int i = 0; i < getChildCount(); ++i) {
            getChildAt(i).layout(0, 0, width, height);
       }

@UserSty
Copy link

UserSty commented Jun 15, 2016

Hi!

I use the solution "nam2210" and when I run on the Nexus 7 (2013) I don't get full screen.
Here is a screenshot: http://i65.tinypic.com/ifvlgg.png

How can I get full screen, please?

@pm0733464
Copy link
Contributor

Have you tried the solution that I posted above on March 30?

@UserSty
Copy link

UserSty commented Jun 15, 2016

Have you tried the solution that I posted above on March 30?

A preview is displayed in the center? It is not attached to the top left corner?

@ggirard07
Copy link

@pm0733464 Can you confirm that using @Nam2210 solution will result in less efficient detection when running on a device having a poor display (low density) but a high res camera?
Especially due to .setRequestedPreviewSize(metrics.heightPixels, metrics.widthPixels)
I know my assumption will probably be false in reality based on latest devices specs. But with the large Android fragmentation, looks like a bad idea to me to base the size of the frames used for the detection on the screen density...

@pm0733464
Copy link
Contributor

@pm0733464 Can you confirm that using @Nam2210 solution will result in less efficient detection when running on a device having a poor display (low density) but a high res camera?

For the face API, using a higher resolution is usually slower than using a lower resolution... slower but sometimes a little more accurate. It's a speed / accuracy trade-off.

Looks like @Nam2210's solution matches the camera preview resolution to the display resolution. Assuming that the display resolution isn't very low (e.g., less than 320x240), it probably isn't less efficient in the low density / high res camera scenario. That is, using the display resolution would probably result in a faster detection than using the higher res camera resolution.

But it's a bit of a different story for the barcode API, since the barcode API tends to require higher resolutions for greater accuracy. In that case, you might want to go with a higher resolution than the display resolution if the display resolution is too low.

@ggirard07
Copy link

@pm0733464 sorry I forgot to specify my comment was for barcode detection indeed :)

@UserSty
Copy link

UserSty commented Jun 15, 2016

@pm0733464 tell me please

Have you tried the solution that I posted above on March 30?

A preview is displayed in the center? It is not attached to the top left corner?

@ggirard07
Copy link

@UserSty having the preview centered on the display then cropped a little on the borders make more sense than having it "pinned" on the top left corner then loosing a bigger part on the right|bottom borders of the preview.
Your users are going to point the camera onto the "object" they want the detection to happen. Naturally, they are going to place it in the middle of the screen. In that case, the @pm0733464 's solution simply makes more sense ;)

@UserSty
Copy link

UserSty commented Jun 15, 2016

@ggirard07 Thanks

@VaibhavDev111
Copy link

Thank You @abhirav . This really helped me. 👍

@UserSty
Copy link

UserSty commented Aug 5, 2016

The solution "@pm0733464" does not work correctly on some devices. The same code snippet is used in the "googly-eyes" example. And there is also a problem.
User "@ggirard07" wrote that the preview is centered on the screen, but it is not. Apparently there is a bug.

I checked the code work on some devices. On the "HTC One M8 (Android 6)" and "Motorola Moto G (2nd Gen.) (Android 5)" preview centered, and then the image is displayed correctly on the screen using "centerCrop".
But on the device, "Asus Nexus 7 2013 (Android 6.0.1)" preview is not centered, and is attached to the top. And now "centerCrop" does not work correctly.
Also, there is a problem in the "Samsung Google Galaxy Nexus (Android 4.4.4)" preview is not centered, and already attached to the bottom. And now "centerCrop" does not work correctly.

How do I know when the preview is centered, and when not?
Is this a bug in mobileVison?

Help solve this problem or tell me how to get around it, thank you.

@wax911
Copy link

wax911 commented Aug 5, 2016

@UserSty Hi, could you try the modified barcode I above Modified Barcode Scanner Code and tell me how it goes, from there I might be able to help you.

@UserSty
Copy link

UserSty commented Aug 5, 2016

@wax911 I tried your code "Modified Barcode Scanner Code" He works too, is not quite correct. I checked the code work on some devices.

On the device, "the Nexus 7 (Android 6)" preview is not displayed on the full screen, and is attached to the upper left corner. This is bad unfortunately.

On the "HTC One M8 (Android 6)" preview is not clear which is tied as the picture is displayed as screen size and still look good.

What I've noticed that if I try to put a different size of previews, it is ignored and displayed differently.

This solution does not work as expected.

Any help please.

@wax911
Copy link

wax911 commented Aug 5, 2016

@UserSty damn, that's a shame then, the reason why it's ignoring your preview sizes is because I automatically get the screen resolution and best camera res paired together from the camera source code. Can you try playing around with:

private static SizePair selectSizePair(Camera camera, int desiredWidth, int desiredHeight) { List validPreviewSizes = generateValidPreviewSizeList(camera); return validPreviewSizes.get(0); }

At some point as I was playing around with this I realized you don't always the preview size requested so I had to forcefully return the first item in the list of validPreviewSizes, I you find this function which I explained a few comments back you can debug the solution as run the code and see what preview sizes are available. If you need any help feel free to drop me an email, I would also like to try and fix this issue. Oh and BTW when I tested this on a tablet it would fill up the screen if the display was in a landscape format.

@ggirard07
Copy link

@UserSty Not sure how you are doing your tests... But I retested @pm0733464 solution currently implemented in my app on an "Asus Nexus 7 2013 (Android 6.0.1)", and everything looks perfectly fine to me!

@UserSty
Copy link

UserSty commented Aug 17, 2016

@ggirard07
I noticed that the preview size is not calculated according to the transfer resolution in the method (setRequestedPreviewSize (metrics.heightPixels, metrics.widthPixels)), but according to the full screen resolution.
Those. (metrics.heightPixels) returns in 1786 but the actual size of the screen height is 1920. So the fact that the previews are always calculated in accordance with, full size without considering the bottom NavigationBar.

I think this is wrong.

@ggirard07
Copy link

metrics.heightPixels is only an estimated size that metrics.heightPixels will try to accommodate.
Bu in fact, it will use the closer preview size available based on device camera!!

Please, prior to declare "issues" do your testing properly in order to avoid people to lose their time... Otherwise you will end with not getting anymore help, just like what is happening on your issue here.
If you have questions about code you do not understand or code you are trying to get right, Stackoverflow if your best shot!

@peterhorsley
Copy link

thanks @pm0733464 your code works perfectly on my samsung j3 👍

@mynelis
Copy link

mynelis commented Feb 17, 2017

This is how I fixed it. And it works perfectly in portrait and landscape modes, full screen.

@OverRide
protected void onLayout (boolean changed, int left, int top, int right, int bottom)
{
int width = getWidth();
int height = getHeight();

    if (mCameraSource != null) {
        Size size = mCameraSource.getPreviewSize();
        if (size != null) {
            width = size.getWidth();
            height = size.getHeight();
        }
    }

    int layoutWidth = right - left;
    int layoutHeight = bottom - top;

    // Swap width and height sizes when in portrait, since it will be rotated 90 degrees
    if (isPortraitMode()) {
        int tmp = width;
        //noinspection SuspiciousNameCombination
        width = height;
        height = tmp;
    }

    int childWidth = layoutWidth;
    int childHeight = (int) (((float) layoutWidth / (float) width) * height);

    if (isPortraitMode()) {
        childHeight = layoutHeight;
        childWidth = (int) (((float) layoutHeight / (float) height) * width);
    }

    for (int i = 0; i < getChildCount(); ++i) {
        getChildAt(i).layout(0, 0, childWidth, childHeight);
    }

    try {
        startIfReady();
    }
    catch (SecurityException se) {
    }
}

@gehadfatah
Copy link

gehadfatah commented Feb 12, 2018

This is how I fixed it .first comment this
if (childHeight > layoutHeight) {
childHeight = layoutHeight;
childWidth = (int)(((float) layoutHeight / (float) height) * width);
}
Then
set int childHeight=layoutHieght;
Work in orientation Landscape and Portrait .

@SethuVignesh
Copy link

@abhirav Thanks Abhirav it worked for me

@GaldinoJr
Copy link

The following works, while also preserving the aspect ratio. This will slightly oversize the camera display, requiring cropping along one dimension. Change CameraSourcePreview.onLayout to:

    @Override
    protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
        int previewWidth = 320;
        int previewHeight = 240;
        if (mCameraSource != null) {
            Size size = mCameraSource.getPreviewSize();
            if (size != null) {
                previewWidth = size.getWidth();
                previewHeight = size.getHeight();
            }
        }

        // Swap width and height sizes when in portrait, since it will be rotated 90 degrees
        if (isPortraitMode()) {
            int tmp = previewWidth;
            previewWidth = previewHeight;
            previewHeight = tmp;
        }

        final int viewWidth = right - left;
        final int viewHeight = bottom - top;

        int childWidth;
        int childHeight;
        int childXOffset = 0;
        int childYOffset = 0;
        float widthRatio = (float) viewWidth / (float) previewWidth;
        float heightRatio = (float) viewHeight / (float) previewHeight;

        // To fill the view with the camera preview, while also preserving the correct aspect ratio,
        // it is usually necessary to slightly oversize the child and to crop off portions along one
        // of the dimensions.  We scale up based on the dimension requiring the most correction, and
        // compute a crop offset for the other dimension.
        if (widthRatio > heightRatio) {
            childWidth = viewWidth;
            childHeight = (int) ((float) previewHeight * widthRatio);
            childYOffset = (childHeight - viewHeight) / 2;
        } else {
            childWidth = (int) ((float) previewWidth * heightRatio);
            childHeight = viewHeight;
            childXOffset = (childWidth - viewWidth) / 2;
        }

        for (int i = 0; i < getChildCount(); ++i) {
            // One dimension will be cropped.  We shift child over or up by this offset and adjust
            // the size to maintain the proper aspect ratio.
            getChildAt(i).layout(
                    -1 * childXOffset, -1 * childYOffset,
                    childWidth - childXOffset, childHeight - childYOffset);
        }

        try {
            startIfReady();
        } catch (IOException e) {
            Log.e(TAG, "Could not start camera source.", e);
        }
    }

Very nice, work perfectly, Thanks!!

@iRYO400
Copy link

iRYO400 commented Jan 21, 2019

Hi! Can somebody share a full screen vertical only solution?

@stalbeal
Copy link

@gehadfatah the best solution for me, thanks!!

@RobertTadevosyan
Copy link

RobertTadevosyan commented Mar 29, 2020

@OverRide
protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
int width = 320;
int height = 240;
if (mCameraSource != null) {
Size size = mCameraSource.getPreviewSize();
if (size != null) {
width = size.getWidth();
height = size.getHeight();
}
}

    // Swap width and height sizes when in portrait, since it will be rotated 90 degrees
    if (isPortraitMode()) {
        int tmp = width;
        width = height;
        height = tmp;
    }

    final int layoutWidth = right - left;
    final int layoutHeight = bottom - top;

    for (int i = 0; i < getChildCount(); ++i) {
        getChildAt(i).layout(0, 0, layoutWidth, layoutHeight);
    }

    try {
        startIfReady();
    } catch (IOException e) {
        Log.e(TAG, "Could not start camera source.", e);
    }
}

also do not forget to comment this line

mCameraSource = new CameraSource.Builder(context, myFaceDetector)
// .setRequestedPreviewSize(640, 480)
.setFacing(CameraSource.CAMERA_FACING_FRONT)
.setRequestedFps(15.0f)
.build();

Here is the right answer, which is working on portrait mode, I did't check it on landscape mode, but it should work, because I am setting the layout width and height according to the screen sizes.

rombyar added a commit to rombyar/fibamlscan that referenced this issue Aug 5, 2020
@SourabBiswas-developer
Copy link

hi anybody know that how to capure image in this

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests