Android:如何在设置图像视图时检测从图库中选择的图像方向(纵向或横向)?

Posted

技术标签:

【中文标题】Android:如何在设置图像视图时检测从图库中选择的图像方向(纵向或横向)?【英文标题】:Android : How to detect the image orientation (portrait or landscape) picked from gallery while setting on an imageview? 【发布时间】:2012-09-25 11:09:54 【问题描述】:

我正在从画廊(相机相册)挑选的图像视图上设置图像。如果选择的图像是横向的,它会完美显示,但如果图像是纵向模式(即图像是在纵向模式下单击的),它会以 90 度旋转显示图像。现在我试图在设置 imageview 之前找出方向,但所有图像都给出相同的方向和相同的宽度高度。这是我的代码:

Uri selectedImage = intent.getData();
if (selectedImage != null) 
    Bitmap bitmap = MediaStore.Images.Media.getBitmap(this.getContentResolver(), selectedImage);

    int str = new ExifInterface(selectedImage.getPath()).getAttributeInt("Orientation", 1000);
    Toast.makeText(this, "value:" + str, Toast.LENGTH_LONG).show();
    Toast.makeText(this, "width:" + bitmap.getWidth() + "height:" + bitmap.getHeight(), Toast.LENGTH_LONG).show();

【问题讨论】:

谁能帮我我有同样的问题..***.com/questions/28379130/… 【参考方案1】:

使用ExifInterface 旋转图像。使用此方法获取正确的值以从相机旋转捕获的图像。

public int getCameraPhotoOrientation(Context context, Uri imageUri, String imagePath)
    int rotate = 0;
    try 
        context.getContentResolver().notifyChange(imageUri, null);
        File imageFile = new File(imagePath);

        ExifInterface exif = new ExifInterface(imageFile.getAbsolutePath());
        int orientation = exif.getAttributeInt(ExifInterface.TAG_ORIENTATION, ExifInterface.ORIENTATION_NORMAL);

        switch (orientation) 
        case ExifInterface.ORIENTATION_ROTATE_270:
            rotate = 270;
            break;
        case ExifInterface.ORIENTATION_ROTATE_180:
            rotate = 180;
            break;
        case ExifInterface.ORIENTATION_ROTATE_90:
            rotate = 90;
            break;
        

        Log.i("RotateImage", "Exif orientation: " + orientation);
        Log.i("RotateImage", "Rotate value: " + rotate);
     catch (Exception e) 
        e.printStackTrace();
    
    return rotate;

并将这段代码放入Activity结果方法并获取旋转图像的值...

String selectedImage = data.getData();
String[] filePathColumn = MediaStore.Images.Media.DATA;

Cursor cursor = getContentResolver().query(selectedImage, filePathColumn, null, null, null);
cursor.moveToFirst();

int columnIndex = cursor.getColumnIndex(filePathColumn[0]);
filePath = cursor.getString(columnIndex);
cursor.close();

int rotateImage = getCameraPhotoOrientation(MyActivity.this, selectedImage, filePath);

希望这会有所帮助..

【讨论】:

exif.getAttributeInt(ExifInterface.TAG_ORIENTATION, ExifInterface.ORIENTATION_NORMAL);对于任何方向图像,总是返回值 0。 对不起,我犯了一个错误。你的方法对我有用。 data.getData() 返回 Uri 而不是 String,请查看您的代码。 @Deepak 为什么我必须打电话给context.getContentResolver().notifyChange(imageUri, null) 使用上面的代码,如果你不能顺时针方向旋转图像,请在将图像设置为图像视图时添加一行imageView.setRotation(angle);这条线帮助了我。 谢谢,但我可以使用相同的代码来捕获视频吗?【参考方案2】:

这也对我有用:

String[] orientationColumn = MediaStore.Images.Media.ORIENTATION;
Cursor cur = getContentResolver().query(imageUri, orientationColumn, null, null, null);
int orientation = -1;
if (cur != null && cur.moveToFirst()) 
    orientation = cur.getInt(cur.getColumnIndex(orientationColumn[0]));

Matrix matrix = new Matrix();
matrix.postRotate(orientation);

【讨论】:

【参考方案3】:
                      if(bm.getWidth() > bm.getHeight())
                        
                            Bitmap bMapRotate=null;
                            Matrix mat=new Matrix();
                            mat.postRotate(90);
                        bMapRotate = Bitmap.createBitmap(bm, 0, 0,bm.getWidth(),bm.getHeight(), mat, true);
                        bm.recycle();
                        bm=null;
                        imageDisplayView.setImageBitmap(bMapRotate);
                        else
                        imageDisplayView.setImageBitmap(bm);

【讨论】:

在某些设备(如 s3 和 s4)上,即使图像是纵向的,宽度也会大于高度。这就是为什么我们不使用这个解决方案,即使它看起来很干净和完美。 @JesusDimrix 是的,可能是这样。但是如果图像的头部没有方向信息,那么我们可以使用这种方式。 我疯了,试图让这个工作你的方式对我来说足够好,银河系什么?【参考方案4】:

这是我遇到的一个很好的解决方案:>https://***.com/a/34241250/8033090

一线解决方案:

Picasso.with(context).load("http://i.imgur.com/DvpvklR.png").into(imageView);

Picasso.with(context).load("file:" + photoPath).into(imageView);

这将自动检测旋转并将图像放置在正确的方向

Picasso 是一个非常强大的库,用于在您的应用中处理图像,包括: 使用最少内存的复杂图像转换。加载可能需要一秒钟,但我只是在图像视图后面放了一些文本,上面写着“正在加载图像”,并且当图像加载时它会覆盖文本。

【讨论】:

对我不起作用。即使使用 Picasso 或 Glide,在 Galaxy S8 或 Galaxy S9 上拍摄的照片仍会旋转 @DenysLobur 你找到解决方案了吗?我也同意这个。银河s8有问题 嘿@Peter,我想我找到了解决方案。你基本上需要实现这个android-developers.googleblog.com/2016/12/…。三星手机给出 'val rotationConstant = ExifInterface.ORIENTATION_*' 作为某个值,例如 1、3、6 等,而其他手机(在我的例子中是小米)给出 'val rotationConstant = ExifInterface.ORIENTATION_UNDEFINED' 等于 0。然后,知道旋转常数,您可以按照自己喜欢的方式旋转图像,如果不是三星手机,则可以将其旋转到 0【参考方案5】:

使用 kotlin 并考虑新 android 版本中 uri 和 exif 的变化:

 var exif: ExifInterface? = null;
 try 
    when (Build.VERSION.SDK_INT) 
        in Int.MIN_VALUE..24 -> exif = ExifInterface(imageUri.path)
        else -> exif = ExifInterface(getContentResolver().openInputStream(data.extras.get("data")))
     
  catch (e: IOException) 
     e.printStackTrace();
 
 val orientation = exif?.getAttributeInt(ExifInterface.TAG_ORIENTATION, ExifInterface.ORIENTATION_NORMAL)
 bmp = rotateBitmap(bmp, orientation ?: ExifInterface.ORIENTATION_NORMAL)

而rotateBitmap函数是:

un rotateBitmap(bitmap: Bitmap, orientation: Int): Bitmap 
val matrix = Matrix()
when (orientation) 
    ExifInterface.ORIENTATION_NORMAL -> return bitmap
    ExifInterface.ORIENTATION_FLIP_HORIZONTAL -> matrix.setScale(-1f, 1f)
    ExifInterface.ORIENTATION_ROTATE_180 -> matrix.setRotate(180f)
    ExifInterface.ORIENTATION_FLIP_VERTICAL -> 
        matrix.setRotate(180f)
        matrix.postScale(-1f, 1f)
    
    ExifInterface.ORIENTATION_TRANSPOSE -> 
        matrix.setRotate(90f)
        matrix.postScale(-1f, 1f)
    
    ExifInterface.ORIENTATION_ROTATE_90 -> matrix.setRotate(90f)
    ExifInterface.ORIENTATION_TRANSVERSE -> 
        matrix.setRotate(-90f)
        matrix.postScale(-1f, 1f)
    
    ExifInterface.ORIENTATION_ROTATE_270 -> matrix.setRotate(-90f)
    else -> return bitmap

try 
    val bmRotated = Bitmap.createBitmap(bitmap, 0, 0, bitmap.width, bitmap.height, matrix, true)
    bitmap.recycle()
    return bmRotated
 catch (e: OutOfMemoryError) 
    e.printStackTrace()
    return null

【讨论】:

考虑翻转非常重要!【参考方案6】:

在某些安卓设备上使用 EXIF 信息不可靠。 (ExifInterface 方向总是返回 0)

这对我有用:

旋转位图

public static Bitmap rotateBitmap(Context context, Uri photoUri, Bitmap bitmap) 

    int orientation = getOrientation(context, photoUri);
    if (orientation <= 0) 
        return bitmap;
    

    Matrix matrix = new Matrix();
    matrix.postRotate(orientation);
    bitmap = Bitmap.createBitmap(bitmap, 0, 0, bitmap.getWidth(), bitmap.getHeight(), matrix, false);
    return bitmap;

从 MediaStore 获取方向

private static int getOrientation(Context context, Uri photoUri) 

    Cursor cursor = context.getContentResolver().query(photoUri,
            new String[]MediaStore.Images.ImageColumns.ORIENTATION, null, null, null);

    if (cursor.getCount() != 1) 
        cursor.close();
        return -1;
    

    cursor.moveToFirst();
    int orientation = cursor.getInt(0);
    cursor.close();
    cursor = null;
    return orientation;

【讨论】:

它需要 API29 :(【参考方案7】:

另一种解决方案是使用支持库中的 ExifInterface: ExifInterface from support library

【讨论】:

【参考方案8】:

这对我有用:

private String getOrientation(Uri uri)
    BitmapFactory.Options options = new BitmapFactory.Options();
    options.inJustDecodeBounds = true;
    String orientation = "landscape";
    try
        String image = new File(uri.getPath()).getAbsolutePath();
        BitmapFactory.decodeFile(image, options);
        int imageHeight = options.outHeight;
        int imageWidth = options.outWidth;
        if (imageHeight > imageWidth)
            orientation = "portrait";
        
    catch (Exception e)
        //Do nothing
    
    return orientation;

【讨论】:

在某些设备(如 s3 和 s4)上,即使图像是纵向的,宽度也会大于高度。这就是为什么我们不使用这个解决方案,即使它看起来很干净和完美。 使用 enum 或 int 结果会更好看:)

以上是关于Android:如何在设置图像视图时检测从图库中选择的图像方向(纵向或横向)?的主要内容,如果未能解决你的问题,请参考以下文章

Android - 将图像设置为用户配置文件的图像视图

如何从图库中加载图像视图中的图像?

如果从图库中选择图像后,如何设置要查看的图像,调用 onCreate 方法并且它已经在该视图中设置了图像?

如何在 android 中提高 OpenCV 人脸检测性能?

从图库中获取图像以在片段中的图像视图中设置? [复制]

如何在 Android 中检测到人脸后将图像下载到图库?