Andorid 11调用系统裁剪
随着各大厂商对android11的升级推送,现在已经有了一定android11的机型,关于android11的适配网上有很多相关的文章
这里主要强调下android11的分区存储一般情况下遇到的问题。
一般非垃圾清理类app或者没有特殊需求的app,主要在调用图片裁剪会遇到android11的问题。
如小米10:
小米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;
总结:app在Android11系统上强制使用分区存储,私有目录下的文件无法被访问,如果有大量文件的话,要做数据迁移(就是将需要共享的文件从上述的私有目录移到共有目录下),这里网上有很多相关的文章。
题外话:还有如果用到umeng分享的话,在android11的设备上分享图片也会有问题类似的问题,具体可以更新新的umeng shareSdk,友盟已经做了相关适配。
学习时的痛苦是暂时的 未学到的痛苦是终生的