Android 文件选择器

选择本地文件,返回文件地址;原本有一个工具类,就随手粘贴上传了,结果,太坑了,同事使用各种崩溃,自己疯狂被打脸;

真的随着Android版本的变化,真的需要把之前的工具类梳理优化下了。

目录

工具类

[版本适配,主要考虑是Android 10 以上版本](#版本适配,主要考虑是Android 10 以上版本)

机型适配


工具类

java 复制代码
public class FilePickerUtil {

    public static final int PICK_FILE_REQUEST_CODE = 1;
    private static FilePickerListener filePickerListener;

    public interface FilePickerListener {
        void onFilePicked(String filePath);
    }

    public static void pickFile(Activity activity, FilePickerListener listener) {
        filePickerListener = listener;
        Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
        intent.setType("*/*");
        intent.addCategory(Intent.CATEGORY_OPENABLE);
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            intent.putExtra(Intent.EXTRA_ALLOW_MULTIPLE, true);
        }
        activity.startActivityForResult(Intent.createChooser(intent, "Select a file"), PICK_FILE_REQUEST_CODE);
    }

    public static void pickFile(Fragment fragment, FilePickerListener listener) {
        filePickerListener = listener;
        Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
        intent.setType("*/*");
        intent.addCategory(Intent.CATEGORY_OPENABLE);
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            intent.putExtra(Intent.EXTRA_ALLOW_MULTIPLE, true);
        }
        fragment.startActivityForResult(Intent.createChooser(intent, "Select a file"), PICK_FILE_REQUEST_CODE);
    }

    public static void onActivityResult(Context context, int requestCode, int resultCode, @Nullable Intent data) {
        if (requestCode == PICK_FILE_REQUEST_CODE && resultCode == Activity.RESULT_OK) {
            if (data != null) {
                Uri uri = data.getData();
                if (uri != null) {
                    String filePath = getFilePath(context, uri);
                    if (filePickerListener != null) {
                        filePickerListener.onFilePicked(filePath);
                    }
                } else if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT && data.getClipData() != null) {
                    // Handle multiple files selected
                    for (int i = 0; i < data.getClipData().getItemCount(); i++) {
                        Uri fileUri = data.getClipData().getItemAt(i).getUri();
                        String filePath = getFilePath(context, fileUri);
                        if (filePickerListener != null) {
                            filePickerListener.onFilePicked(filePath);
                        }
                    }
                }
            }
        }
    }

    private static String getFilePath(Context context, Uri uri) {
        Log.d("FilePickerUtil", "URI: " + uri.toString());
        String filePath = null;
        String authority = uri.getAuthority();

        // 定义一个映射表,存储不同品牌文件管理器的 URI 处理逻辑
        Map<String, String> authorityToRootPathMap = new HashMap<>();
        authorityToRootPathMap.put("com.hihonor.filemanager.share.fileprovider", "/root");
        authorityToRootPathMap.put("com.huawei.filemanager.share.fileprovider", "/root");
        authorityToRootPathMap.put("com.mi.android.globalFileexplorer.myprovider", "/root");
        authorityToRootPathMap.put("com.coloros.filemanager.fileprovider", "/root");
        authorityToRootPathMap.put("com.vivo.filemanager.share.fileprovider", "/root");
        // 可以根据需要添加更多品牌的文件管理器

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            if (DocumentsContract.isDocumentUri(context, uri)) {
                String docId = DocumentsContract.getDocumentId(uri);
                String[] split = docId.split(":");
                String type = split[0];
                Uri contentUri = null;
                if ("primary".equalsIgnoreCase(type)) {
                    filePath = context.getExternalFilesDir(null) + "/" + split[1];
                } else 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;
                }
                if (contentUri != null) {
                    filePath = getDataColumn(context, contentUri, "_id=?", new String[]{split[1]});
                }
                Log.d("FilePickerUtil", "DocumentsContract URI: " + filePath);
            }
        }
        if (filePath == null) {
            // 使用映射表处理不同品牌文件管理器提供的 URI
            if (authorityToRootPathMap.containsKey(authority)) {
                String rootPath = authorityToRootPathMap.get(authority);
                filePath = uri.getPath().replace(rootPath, "");
            } else {
                filePath = getDataColumn(context, uri, null, null);
            }
        }
        Log.d("FilePickerUtil", "File Path: " + filePath);
        return filePath;
    }

    private static String getDataColumn(Context context, Uri uri, String selection, String[] selectionArgs) {
        Cursor cursor = null;
        final String[] projection = {"_data", "_display_name", "_size"}; // 移除不存在的列 "data"
        try {
            cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs, null);
            if (cursor != null && cursor.moveToFirst()) {
                for (String column : projection) {
                    int columnIndex = cursor.getColumnIndex(column);
                    if (columnIndex != -1) {
                        return cursor.getString(columnIndex);
                    }
                }
            }
        } finally {
            if (cursor != null)
                cursor.close();
        }
        return null;
    }
}

版本适配,主要考虑是Android 10 以上版本

  • Android 10 及以上版本 强调隐私和安全,要求使用 SAF 来访问文件,并引入了范围存储。
复制代码
  Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT);
  • Android 10 以下版本 访问文件的方式相对简单,应用程序可以直接访问外部存储空间中的文件。
复制代码
  Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
  • 访问外部存储空间的权限要求发生了变化,应用程序需要请求 READ_EXTERNAL_STORAGEWRITE_EXTERNAL_STORAGE 权限来访问公共存储空间。
  • 在 Android 11 及以上版本中,应用程序还需要请求 MANAGE_EXTERNAL_STORAGE 权限来获得对外部存储的完全访问权限。

机型适配

特定的文件管理器应用程序,共享的文件路径。FileProvider 是 Android 中的一种机制,允许应用程序之间安全地共享文件,而不必直接暴露文件的实际路径。 所以在解析URL的时候需要添加机型版本的适配,

是的,每个机型返回的共享文件路径都是不一样。

java 复制代码
// 定义一个映射表,存储不同品牌文件管理器的 URI 处理逻辑
        Map<String, String> authorityToRootPathMap = new HashMap<>();
        authorityToRootPathMap.put("com.hihonor.filemanager.share.fileprovider", "/root");
        authorityToRootPathMap.put("com.huawei.filemanager.share.fileprovider", "/root");
        authorityToRootPathMap.put("com.mi.android.globalFileexplorer.myprovider", "/root");
        authorityToRootPathMap.put("com.coloros.filemanager.fileprovider", "/root");
        authorityToRootPathMap.put("com.vivo.filemanager.share.fileprovider", "/root");
        // 可以根据需要添加更多品牌的文件管理器


       // 使用映射表处理不同品牌文件管理器提供的 URI
            if (authorityToRootPathMap.containsKey(authority)) {
                String rootPath = authorityToRootPathMap.get(authority);
                filePath = uri.getPath().replace(rootPath, "");
            } else {
                filePath = getDataColumn(context, uri, null, null);
            }

我是做了一个映射表,目前只有 荣耀、小米、华为、vivo、oppo;

相关推荐
zhuiQiuMX13 分钟前
笔试阶段性心得总结
java·python
星沁城1 小时前
236. 二叉树的最近公共祖先
java·数据结构·leetcode·二叉树
oliveira-time2 小时前
Java 1.8(也称为Java 8)
java·开发语言
极小狐4 小时前
如何使用极狐GitLab 软件包仓库功能托管 maven?
java·运维·数据库·安全·c#·gitlab·maven
.生产的驴4 小时前
SpringBoot 集成滑块验证码AJ-Captcha行为验证码 Redis分布式 接口限流 防爬虫
java·spring boot·redis·分布式·后端·爬虫·tomcat
野犬寒鸦6 小时前
MySQL索引使用规则详解:从设计到优化的完整指南
java·数据库·后端·sql·mysql
思考的橙子6 小时前
Springboot之会话技术
java·spring boot·后端
钰爱&6 小时前
【Linux】POSIX 线程信号量与互斥锁▲
java·开发语言·jvm
yt948327 小时前
Matlab实现绘制任意自由曲线
开发语言·matlab
黑匣子~8 小时前
java集成telegram机器人
java·python·机器人·telegram