Andorid 11调用系统裁剪

随着各大厂商对android11的升级推送,现在已经有了一定android11的机型,关于android11的适配网上有很多相关的文章
这里主要强调下android11的分区存储一般情况下遇到的问题。
一般非垃圾清理类app或者没有特殊需求的app,主要在调用图片裁剪会遇到android11的问题。
如小米10:

  Andorid 11调用系统裁剪 小米10裁剪.jpg
  Andorid 11调用系统裁剪 小米10裁剪报错.jpg
小米10报错:保存时发生错误,保存失败
纠其原因就是android11在更新后,会强制使用分区存储:
在tagSdk<30(29未忽略分区存储情况下requestLegacyExternalStorage=true)其他应用,无法访问app私有目录下的文件;所以导致了上图出现系统裁剪应用,无法访问app下的裁剪副本,这里就讲到了系统裁剪功能流程:
1:在拍照或者相册中拿到图片的uri
2:创建图片intent传入ImageType:intent.setDataAndType(uri, "image/*")
3:创建裁剪输出路径:intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.parse("file://"+ mOnputFile.getAbsolutePath()));
4:裁剪完成后的图片文件就是mOnputFile
这里在android11上出错就是在第3步骤上,内置系统裁剪应用无法访问自己的app私有目录下的图片(这里大多数应用在访问图片文件的时候应该都是采用fileprovider创建的私有目录的uri吧)
我们应用的文件存储的内置路径是:
storage/emulated/0/Android/data/com.xx.xxxx/cache
可以看到com.xx.xxxx就是内置的私有目录
所以,在android11上,通过fileprovider创建的uri path只要改为公域,系统裁剪应用就可以访问裁剪过后,公域图片地址:
storage/emulated/0/Pictures
可以看到没有包名。
获取公域地址方法:

 

String path = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES).getPath()

完整代码,这里以相册为例:

    // 启动相册
    public void openAlbum() {
        Intent intent_album = new Intent("android.intent.action.GET_CONTENT");
        intent_album.setType("image/*");
        startActivityForResult(intent_album, ICON_FROM_ALBUM);
    }

启动相册后,在onActivityResult里获取uri

  case ICON_FROM_ALBUM:
        //从相册选择
      if (data == null || data.getData() == null) {
        return;
      }
      clipPhoto(Uri.fromFile(new File(ImageFileUtils.getPath(this, data.getData()))
   )); //裁剪图片
 break;

图片裁剪,这里在tagSdk低于30的情况下做判断,注意7.0的通过provider的uri来访问媒体

 //裁剪图片
    private void clipPhoto(Uri uri) {
        Intent intent = new Intent("com.android.camera.action.CROP");
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            uri = FileProvider.getUriForFile(this, BuildConfig.APPLICATION_ID + ".fileprovider", new File(uri.getPath()));
            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
        }
        intent.setDataAndType(uri, "image/*");
        // 下面这个crop=true是设置在开启的Intent中设置显示的VIEW可裁剪
        intent.putExtra("crop", "true");
        // aspectX aspectY 是宽高的比例
        intent.putExtra("aspectX", 1);
        intent.putExtra("aspectY", 1);
        // outputX outputY 是裁剪图片宽高
        intent.putExtra("outputX", 150);
        intent.putExtra("outputY", 150);
        intent.putExtra("circleCrop", true);
        if (Build.VERSION.SDK_INT >= 30) {
            //android 11以上,将文件创建在公有目录
            String path = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES).getPath();
            //storage/emulated/0/Pictures
            mOnputFile = new File(path, System.currentTimeMillis() + ".png");
            intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.parse("file://" + mOnputFile.getAbsolutePath()));
        } else {
            //storage/emulated/0/Android/data/com.xxxxx/cache
            mOnputFile = new File(sdPath, System.currentTimeMillis() + ".png");
            intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.parse("file://" + mOnputFile.getAbsolutePath()));
        }
        startActivityForResult(intent, ICON_CROP);
    }

依然在onActivityResult中处理

case ICON_CROP:
  //mOnputFile就是裁剪副本,这里简单地判断下文件,可以处理裁剪图片的显示,或者上传
    break;

 

Andorid 11调用系统裁剪 成功.jpg

总结:app在Android11系统上强制使用分区存储,私有目录下的文件无法被访问,如果有大量文件的话,要做数据迁移(就是将需要共享的文件从上述的私有目录移到共有目录下),这里网上有很多相关的文章。

题外话:还有如果用到umeng分享的话,在android11的设备上分享图片也会有问题类似的问题,具体可以更新新的umeng shareSdk,友盟已经做了相关适配。

demo地址

     
上一篇:04-Nginx配置实例-反向代理


下一篇:http协议:四 (4)http的重定向和跳转