升级SDK30之后,一步一个坑。走不完的路,填不完的坑。
Android11存储机制变更:https://developer.android.com/about/versions/11/privacy/storage?hl=zh-cn
一、遇到的问题
调用系统相册选取照片,然后裁剪,存储裁剪之后的照片,显示并上传服务器。多么正常的操作流程,相信大家都已经用过或者正在用。SDK升级到30,在Android11 上就失灵了。What?都知道是存储问题,怎么解决。现在告诉你。
二、解决思路
1、申请MANAGE_EXTERNAL_STORAGE最高的文件读取权限(一般不用)
2、通过 MediaStore API操作(我的选择)
1、SDK>=30 因为系统裁剪不能访问App的私有路径,
所以File 保存到公有路径Environment.getExternalStoragePublicDirectory(DIRECTORY_PICTURES)
2、SDK<30 可以保存到App的私有路径(App卸载可以删除) context.getExternalFilesDir(null).getAbsoluteFile()
三、实现
1、onActivityResult 回调
if (requestCode == REQUEST_CODE_ALBUM) {
if (data != null && data.getData() != null) {
//打开系统裁剪
gotoCrop(data.getData());
}
} else if (resultCode == RESULT_OK && requestCode == REQUEST_CODE_CAPTURE_CROP) {
//显示页面上
if (imageCropFile != null && imageCropFile.getAbsolutePath() != null) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) {
if (FileUtils.uri != null) {
// 通过存储的uri 查询File
imageCropFile = FileUtils.getCropFile(this, FileUtils.uri);
GlideUtils.loadLocalImage(this, FileUtils.uri, -1, iv_avatar);
}
} else {
GlideUtils.loadLocalImage(this, imageCropFile.getAbsolutePath(), -1, iv_avatar);
}
}
}
2、打开系统相册
Intent intent = new Intent(Intent.ACTION_PICK, android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
startActivityForResult(intent, REQUEST_CODE_ALBUM);
3、打开系统裁剪
private void gotoCrop(Uri sourceUri) {
imageCropFile = FileUtils.createImageFile(this, true);
if (imageCropFile != null) {
Intent intent = new Intent("com.android.camera.action.CROP");
intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
intent.putExtra("crop", "true");
intent.putExtra("aspectX", 1); //X方向上的比例
intent.putExtra("aspectY", 1); //Y方向上的比例
intent.putExtra("outputX", 256); //裁剪区的宽
intent.putExtra("outputY", 256); //裁剪区的高
intent.putExtra("scale ", true); //是否保留比例
intent.putExtra("return-data", false);
intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
intent.setDataAndType(sourceUri, "image/*"); //设置数据源
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) {
intent.putExtra(MediaStore.EXTRA_OUTPUT, FileUtils.uri);
} else {
Uri imgCropUri = Uri.fromFile(imageCropFile);
intent.putExtra(MediaStore.EXTRA_OUTPUT, imgCropUri);
}
startActivityForResult(intent, REQUEST_CODE_CAPTURE_CROP);
}
}
4、生成本地File(重要)
public static File getAppRootDirPath() {
return GlobalService.getAppContext().getExternalFilesDir(null).getAbsoluteFile();
}
public static Uri uri;
public static File createImageFile(Context context,boolean isCrop) {
try {
String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss").format(new Date());
String fileName = "";
if (isCrop) {
fileName = "IMG_"+timeStamp+"_CROP.jpg";
} else {
fileName = "IMG_"+timeStamp+".jpg";
}
File rootFile = new File(getAppRootDirPath() + File.separator + "capture");
if (!rootFile.exists()) {
rootFile.mkdirs();
}
File imgFile;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.R) {
imgFile = new File(Environment.getExternalStoragePublicDirectory(DIRECTORY_PICTURES) + File.separator + fileName);
// 通过 MediaStore API 插入file 为了拿到系统裁剪要保存到的uri(因为App没有权限不能访问公共存储空间,需要通过 MediaStore API来操作)
ContentValues values = new ContentValues();
values.put(MediaStore.Images.Media.DATA, imgFile.getAbsolutePath());
values.put(MediaStore.Images.Media.DISPLAY_NAME, fileName);
values.put(MediaStore.Images.Media.MIME_TYPE, "image/jpeg");
uri = context.getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);
}else {
imgFile = new File(rootFile.getAbsolutePath() + File.separator + fileName);
}
return imgFile;
} catch (Exception e) {
e.printStackTrace();
return null;
}
}
5、通过存储的Uri获取File
是不是有疑问 imageCropFile 这个File干啥的 --- 我用来上传服务器用的
public static File getCropFile(Context context,Uri uri){
String[] proj = { MediaStore.Images.Media.DATA };
Cursor cursor = context.getContentResolver().query(uri, proj, null, null, null);
if (cursor.moveToFirst()) {
int columnIndex = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);
String path = cursor.getString(columnIndex);
cursor.close();
return new File(path);
}
return null;
}
仅供参考。如有建议和意见,请及时沟通。