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;

相关推荐
louisgeek2 分钟前
Android 类加载机制
android
SoFlu软件机器人2 分钟前
高并发秒杀系统设计:关键技术解析与典型陷阱规避
java·人工智能
码农小站6 分钟前
MyBatis-Plus 表字段策略详解:@TableField(updateStrategy) 的配置与使用指南
java·后端
李憨憨12 分钟前
深入探究MapStruct:高效Java Bean映射工具的全方位解析
java·后端
雷渊12 分钟前
通俗易懂的来解释倒排索引
java·后端·面试
知其然亦知其所以然12 分钟前
面试官狂喜!我用这 5 分钟讲清了 ThreadPoolExecutor 饱和策略,逆袭上岸
java·后端·面试
碎风,蹙颦13 分钟前
Android开发过程中遇到的SELINUX权限问题
android·人工智能
洛小豆19 分钟前
NoClassDefFoundError 和 ClassNotFoundException 有什么区别?
java·后端·面试
XuanXu20 分钟前
GraalVM的黑科技个人尝鲜总结
java
百锦再20 分钟前
Python实现浏览器模拟访问及页面解析的全面指南
开发语言·前端·javascript·python·vue·框架·react