android:调用原生的activity(即:ACTION_GET_CONTENT)进行文件选择,实现多个文件一起选择

前几天做一个东西,需要从手机选择文件。我选择最简单的调用android自带的选择文件activity进行操作,这里做一下总结。
这里简单把实现的过程分为三个步骤:

  1. 打开选择文件activity
  2. 拿到选到文件的uri资源
  3. 将uri转换为对应的path,在path转化为File,就拿到对应的文件了
    首先是打开选择文件的activity,使用Intent显示调用,下面为对应的代码:
Intent chooseFile = new Intent(Intent.ACTION_GET_CONTENT);
chooseFile.setType("*/*");//匹配所有的类型
//intent.setType(“image/*”);//选择图片
//intent.setType(“audio/*”); //选择音频
//intent.setType(“video/*”); //选择视频 (mp4 3gp 是android支持的视频格式)
//intent.setType(“video/*;image/*”);//同时选择视频和图片
chooseFile.putExtra(Intent.EXTRA_ALLOW_MULTIPLE, true);//设置可以多选文件
Intent intent = Intent.createChooser(chooseFile, "title");
startActivityForResult(intent, 1);

可以看到我们是使用startActivityForResult(intent, 1);的方式打开的该activity,所有当使用这个界面选择完成后,该界面会像我们这个activity反馈选择到的数据。重写onActivityResult(int requestCode, int resultCode, Intent data)方法,对返回数据进行处理和接收,首先判断resultCode==RESULT_OK,然后我们判断requestCode(即:startActivityForResult(intent, 1)的第二个参数,也就是1),进行对应的操作。
接下来就是在拿到对应的uri,这里有一个小小的坑,笔者也是一个一个找了半天。如果我们只选择一个文件,默认不加chooseFile.putExtra(Intent.EXTRA_ALLOW_MULTIPLE, true)这句话就只能选择一个文件,那么这时候得到uri的方法就很简单,只需要Uri uri = data.getData();就可以了。但是很多时候这样是不满足需求的,就需要一下子选择多个文件,那么这时候的到uri就需要这样做了

ClipData clipData = data.getClipData();
if (clipData!=null)
 {
for (int i=0;i<clipData.getItemCount();i++)
 {
ClipData.Item itemAt = clipData.getItemAt(i);
 Uri uri = itemAt.getUri();
 }
 }

这个时候有人可能就说,那我直接解析ClipData 对象不就行了吗?这里有一个大坑,就是如果只选择一个文件那么ClipData为null,只能使用data.getData();但是如果选择多个文件的话,data.getData()得到的值又变成null了,所以如果想使用ACTION_GET_CONTENT实现多文件上传,必须这两种都存在,且判断不为null的情况下得到uri。
再说通过uri得到对应文件的地址(path),这个我直接放一个工具类,将uri转换为path这里就不再特别说明了
工具类

package com.epoint.app.util;

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;

/**
 * 作者: ZXF
 * 创建时间: 2020/8/18 21:29
 * 版本: [1.0, 2020/8/18]
 * 描述: <通过uri得到文件的路径>
 */
public class FileUtils {
    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];
                }

                // TODO handle non-primary volumes
            }
            // 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;
    }
    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;
    }
    public static boolean isExternalStorageDocument(Uri uri) {
        return "com.android.externalstorage.documents".equals(uri.getAuthority());
    }
    public static boolean isDownloadsDocument(Uri uri) {
        return "com.android.providers.downloads.documents".equals(uri.getAuthority());
    }
    public static boolean isMediaDocument(Uri uri) {
        return "com.android.providers.media.documents".equals(uri.getAuthority());
    }
    public static boolean isGooglePhotosUri(Uri uri) {
        return "com.google.android.apps.photos.content".equals(uri.getAuthority());
    }
}

下面放一个选择文件后结果的处理:File只是拿到了,大家可以根据需求使用集合,对file进行储存和上传等等。

protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        if (resultCode==RESULT_OK)
        {
        		switch(requestCode)
            {
                 case 1 :
							//使用ACTION_GET_CONTENT时 选择文件时多个调用用ClipData
                        ClipData clipData = data.getClipData();
                        if (clipData!=null)
                        {
                            for (int i=0;i<clipData.getItemCount();i++)
                            {
                                ClipData.Item itemAt = clipData.getItemAt(i);
                                String path1 = FileUtils.getPath(this,itemAt.getUri());//使用工具类对uri进行转化
                                File file = new File(path1);
                            }
                        }
                        //选择文件单个直接使用Uri
                        Uri uri = data.getData();
                        if (uri!=null)
                        {
                            String path1 = FileUtils.getPath(this,uri);//使用工具类对uri进行转化
                            File file = new File(path1);
                        }
				break;
            }
        }
}
Logo

华为开发者空间,是为全球开发者打造的专属开发空间,汇聚了华为优质开发资源及工具,致力于让每一位开发者拥有一台云主机,基于华为根生态开发、创新。

更多推荐