android 11 拍照裁剪适配

项目用到了app拍照相册裁剪,因为测试机是android10的,一顿操作猛如虎,功能完美呀。然后测试小姐姐用android11测试,直接GG了。
看了之后才发现android11强制分区处理。
android 适配拍照主要经历了 6.0 7.0 10和11
6.0主要是处理运行时权限 7.0为FileProvider 文件共享 10和11则是分区处理。
主要说下7.0和10,11的适配方法。

参考:鸿洋_: Android 7.0 行为变更 通过FileProvider在应用间共享文件吧
官方文档:Android 7.0 行为变更
android7.0 主要是适配其中文件访问权限

在官方7.0的以上的系统中,尝试传递 file://URI可能会触发FileUriExposedException。

清单文件配置provider ,因为它是ContentProvider的子类。

      
            
        

android:authorities 表示授权者,这里的格式一般是[appId].fileprovider
android:exported 只能为false
android:grantUriPermissions="true" 表示授权Uri权限 ,且必须为true
meta-data里设置指定的文件目录,为引用provider_paths文件

provider_paths文件


    
    
    
    
    
     

代表设备的根目录new File("/");
代表context.getFilesDir()
代表context.getCacheDir()
代表Environment.getExternalStorageDirectory()
代表context.getExternalFilesDirs()
代表getExternalCacheDirs()
常用到的 为 external-path 和external-files-path name和path按照自己需求编写。

private void openCamera() {
        Intent captureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        // 判断是否有相机
        if (captureIntent.resolveActivity(getPackageManager()) != null) {
            File photoFile = null;
            try {
               photoFile =  new File(getExternalCacheDir(), "head_image.jpg");
            } catch (IOException e) {
                e.printStackTrace();
             }
            if (photoFile != null) {
               //根据判断版本 4.4也能安全运行 或者 授权
                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                    //适配Android 7.0文件权限,通过FileProvider创建一个content类型的Uri  主要是这段代码
                    photoUri = FileProvider.getUriForFile(this, getPackageName() + ".fileProvider", photoFile);
                } else {
                    photoUri = Uri.fromFile(photoFile);
                }
            }
            if (photoUri != null) {
                Log.e("TAG", "openCamera: 拍照初始路径===" + photoUri);
                captureIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoUri);
                captureIntent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
                startActivityForResult(captureIntent, CODE_CAMERA_REQUEST);
            }
        }
    }

7.0大致就是这样。
android10 如果 targetSdkVersion<=29 在清单文件中添加

  android:requestLegacyExternalStorage="true"

可以让android10忽略分区,按照之前的方式。但是运行在android11的机型就会强制执行分区处理,所以这个只是过度方式,还是得适配。
我遇到得问题就是一个拍照裁剪------加载图片失败,追其原因,为系统裁剪不能访问到app得私有目录,所以我们
主要进行一个裁剪公域得处理。

public static Intent getCropIntent(Uri uri,Context context,String path){
        Log.d("裁剪的Url", "cropRawPhoto: " + uri.toString());
        Intent intent = new Intent("com.android.camera.action.CROP");
        intent.setDataAndType(uri, "image/*");
        intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION | Intent.FLAG_GRANT_READ_URI_PERMISSION);
        // 设置裁剪
        intent.putExtra("crop", "true");
        intent.putExtra("aspectX", 1);
        intent.putExtra("aspectY", 1);
        intent.putExtra("outputX", output_X);
        intent.putExtra("outputY", output_Y);
        intent.putExtra("return-data", false);
        context.getExternalFilesDir(Environment.DIRECTORY_PICTURES);
        //android11 分区存储
        if (Build.VERSION.SDK_INT >= 29) {
            Log.e("TAG", "裁剪公域::"+path);
            File mOnputFile11 = new File(path, SpUtils.getInstance(context).getString("token","") + ".png");
            intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.parse("file://" + mOnputFile11.getAbsolutePath()));
        } else {
            intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(
                    new File(context.getExternalCacheDir(), "face-cropped.jpg")));
        }
        return intent;
    }

//等会儿。

你可能感兴趣的:(android 11 拍照裁剪适配)