android的7.0系统Uri适配读取

这里以选择手机相册为例子,

1、打开手机相册:

```

public static void localPhone(Activity baseActivity) {

        Intent intent = new Intent(

                Intent.ACTION_PICK,

                MediaStore.Images.Media.EXTERNAL_CONTENT_URI);

        intent.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, "image/*");

        baseActivity.startActivityForResult(intent, G.LOCAL_PHONE);// //适用于4.4及以上android版本

    }

```

2、在Activity的onActivityResult处理:

```

@Override

    protected void onActivityResult(int requestCode, int resultCode, Intent data) {

        super.onActivityResult(requestCode, resultCode, data);

        if (resultCode == RESULT_OK) { // 从相册中选择返回

            if (requestCode == G.LOCAL_PHONE) {

                headImageFile = ImageTools.getCropImageFilePath(this);

                ImageTools.startPhotoZoom(HomeActivity.this, data.getData(), headImageFile);

            }

            if (requestCode == G.CRIP_PHONE) { // 从裁剪后返回

                Bundle bundle = data.getExtras();

                if (bundle != null) {

                    //在这里获得了剪裁后的Bitmap对象,可以用于上传

                    Bitmap image = bundle.getParcelable("data");

                    home_im_label.setImageBitmap(image);

                    Log.d("裁剪路径:", headImageFile.getAbsolutePath());

                }

            }

        }

    }

```

代码中startPhotoZoom的方法:

```

/**

    * 裁剪图片方法实现

    * 拍照调用startPhotoZoom(tempUri,this);

    * 从本地选取调用startPhotoZoom(data.getData(),this);

    *

    * @param uri

    */

    public static void startPhotoZoom(Activity baseActivity, Uri uri,File outFile) {

        if (uri == null) {

            Log.i("startPhotoZoom", "The uri is not exist.");

        }

        Intent intent = new Intent("com.android.camera.action.CROP");

        File uriFile = new File(TTImageUtil.getPath(baseActivity, uri));

        Log.i("startPhotoZoom", "uriFile path:" + uriFile.getAbsolutePath());

        Uri imageUri;

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {

            imageUri = FileProvider.getUriForFile(baseActivity, BuildConfig.APPLICATION_ID + ".fileProvider", uriFile);

        } else {

            imageUri = Uri.fromFile(uriFile);

        }

        intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);

        intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

        intent.setDataAndType(imageUri, "image/*");

        // crop为true是设置在开启的intent中设置显示的view可以剪裁

        intent.putExtra("crop", "true");

        intent.putExtra("scale", true);

        // aspectX aspectY 是宽高的比例

        intent.putExtra("aspectX", 1);

        intent.putExtra("aspectY", 1);

        // outputX,outputY 是剪裁图片的宽高

        intent.putExtra("outputX", 150);

        intent.putExtra("outputY", 150);

        intent.putExtra("return-data", false);

        // 取消人脸识别

        intent.putExtra("noFaceDetection", true);

        intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());

        intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(outFile));//定义输出的File Uri

        baseActivity.startActivityForResult(intent, G.CRIP_PHONE);

    }


```

getCropImageFilePath方法的代码:

```

/**

    * 裁剪后,把图片放在这里

    * @param context

    * @return

    */

    public static File getCropImageFilePath(Context context) {

        File imageFile = new File(context.getExternalFilesDir(""), "cropImage.jpg");

        try {

            if (imageFile.exists()) {

                imageFile.delete();

            }

            imageFile.createNewFile();

        } catch (IOException e) {

            e.printStackTrace();

        }

        return imageFile;

    }

```

注意7.0以上的手机系统为了适配Uri.fromFile的方法,必须配置FileProvider,

1. 在主项目的res新建文件夹为xml,并添加file_provider_paths.xml在xml文件夹下,如图:![-w329](media/15694057488260/15694622009737.jpg)

file_provider_paths.xml的内容:

```

<?xml version="1.0" encoding="utf-8"?>

<paths>

    <external-path

        name="external_hsyg"

        path="." />

    <cache-path

        name="update_cache"

        path="." />

</paths>

```

2.在AndroidMainfest.xml配置provider,application的节点内添加如下代码:

```

<provider

            android:name="android.support.v4.content.FileProvider"

            android:authorities="com.example.employee.fileProvider"

            android:exported="false"

            android:grantUriPermissions="true"

            tools:replace="android:authorities">

            <meta-data

                android:name="android.support.FILE_PROVIDER_PATHS"

                android:resource="@xml/file_provider_paths"

                tools:replace="android:resource" />

        </provider>

```

如果是用了androidX的话,就不能引用support包,要像如下:

```

<provider

                android:name="androidx.core.content.FileProvider"

                android:authorities="com.org.wms.fileprovider"

                android:exported="false"

                android:grantUriPermissions="true">

            <meta-data

                    android:name="android.support.FILE_PROVIDER_PATHS"

                    android:resource="@xml/file_provider_paths"/>

        </provider>

```

那就大功告成了!

最后,注意下从相册选择图片回来后的Uri路径的读取,4.4以上系统要做特别的处理,上面的代码已经做了,现在贴最后的代码供参考:

```

import android.annotation.SuppressLint;

import android.content.ContentUris;

import android.content.Context;

import android.database.Cursor;

import android.net.Uri;

import android.os.Build;

import android.os.Environment;

import android.provider.DocumentsContract;

import android.provider.MediaStore;

class TTImageUtil {

    @SuppressLint("NewApi")

    public static String getPath(final Context context, final Uri uri) {

        final boolean isKitKat = Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT;

        // DocumentProvider

        if (isKitKat && DocumentsContract.isDocumentUri(context, uri)) {

            // ExternalStorageProvider

            if (isExternalStorageDocument(uri)) {

                final String docId = DocumentsContract.getDocumentId(uri);

                final String[] split = docId.split(":");

                final String type = split[0];

                if ("primary".equalsIgnoreCase(type)) {

                    return Environment.getExternalStorageDirectory() + "/" + split[1];

                }

            }

            // DownloadsProvider

            else if (isDownloadsDocument(uri)) {

                final String id = DocumentsContract.getDocumentId(uri);

                final Uri contentUri = ContentUris.withAppendedId(

                        Uri.parse("content://downloads/public_downloads"), Long.valueOf(id));

                return getDataColumn(context, contentUri, null, null);

            } // MediaProvider

            else if (isMediaDocument(uri)) {

                final String docId = DocumentsContract.getDocumentId(uri);

                final String[] split = docId.split(":");

                final String type = split[0];

                Uri contentUri = null;

                if ("image".equals(type)) {

                    contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;

                } else if ("video".equals(type)) {

                    contentUri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI;

                } else if ("audio".equals(type)) {

                    contentUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;

                }

                final String selection = "_id=?";

                final String[] selectionArgs = new String[] {

                        split[1]

                };

                return getDataColumn(context, contentUri, selection, selectionArgs);

            }

        }

        // MediaStore (and general)

        else if ("content".equalsIgnoreCase(uri.getScheme())) {

            // Return the remote address

            if (isGooglePhotosUri(uri))

                return uri.getLastPathSegment();

            return getDataColumn(context, uri, null, null);

        }

        // File

        else if ("file".equalsIgnoreCase(uri.getScheme())) {

            return uri.getPath();

        }

        return null;

    }

    /**

    * Get the value of the data column for this Uri. This is useful for

    * MediaStore Uris, and other file-based ContentProviders.

    *

    * @param context The context.

    * @param uri The Uri to query.

    * @param selection (Optional) Filter used in the query.

    * @param selectionArgs (Optional) Selection arguments used in the query.

    * @return The value of the _data column, which is typically a file path.

    */

    public static String getDataColumn(Context context, Uri uri, String selection,

                                      String[] selectionArgs) {

        Cursor cursor = null;

        final String column = "_data";

        final String[] projection = {

                column

        };

        try {

            cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs,

                    null);

            if (cursor != null && cursor.moveToFirst()) {

                final int index = cursor.getColumnIndexOrThrow(column);

                return cursor.getString(index);

            }

        } finally {

            if (cursor != null)

                cursor.close();

        }

        return null;

    }

    /**

    * @param uri The Uri to check.

    * @return Whether the Uri authority is ExternalStorageProvider.

    */

    public static boolean isExternalStorageDocument(Uri uri) {

        return "com.android.externalstorage.documents".equals(uri.getAuthority());

    }

    /**

    * @param uri The Uri to check.

    * @return Whether the Uri authority is DownloadsProvider.

    */

    public static boolean isDownloadsDocument(Uri uri) {

        return "com.android.providers.downloads.documents".equals(uri.getAuthority());

    }

    /**

    * @param uri The Uri to check.

    * @return Whether the Uri authority is MediaProvider.

    */

    public static boolean isMediaDocument(Uri uri) {

        return "com.android.providers.media.documents".equals(uri.getAuthority());

    }

    /**

    * @param uri The Uri to check.

    * @return Whether the Uri authority is Google Photos.

    */

    public static boolean isGooglePhotosUri(Uri uri) {

        return "com.google.android.apps.photos.content".equals(uri.getAuthority());

    }

}

```

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,491评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,856评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,745评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,196评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,073评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,112评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,531评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,215评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,485评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,578评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,356评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,215评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,583评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,898评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,497评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,697评论 2 335

推荐阅读更多精彩内容