第一步:FileProvider相关准备工作
在AndroidManifest.xml中增加provider节点:
<provider
android:name="androidx.core.content.FileProvider"
android:authorities="com.choosecrop.fileprovider"
android:grantUriPermissions="true"
android:exported="false">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/filepaths" />
</provider>
其中:
android:authorities 表示授权列表,填写你的应用包名,当有多个授权时,用分号隔开。
android:exported 表示该内容提供器(FileProvider)是否能被第三方程序组件使用,必须为false,否则会报异常: java.lang.SecurityException: Provider must not be exported。
android:grantUriPermissions=“true” 表示授予 URI 临时访问权限。
接着,需要在res目录下建立一个xml目录,
<?xml version="1.0" encoding="utf-8"?>
<paths>
<external-files-path name="DCIM" path="." />
</paths>
说明:
files-path name=“name” path=“path” 对应 Context.getFilesDir()
cache-path name=“name” path=“path” 对应 getCacheDir()
external-path name=“name” path=“path” 对应Environment.getExternalStorageDirectory()
external-files-path name=“name” path=“path” 对应 Context.getExternalFilesDir()
external-cache-path name=“name” path=“path” 对应 Context.getExternalCacheDir() external-media-path name=“name” path=“path” 对应 Context.getExternalMediaDirs()
注意:ExternalStorageDirectory在29后废弃,官方推荐使用Context.getExternalFilesDir()
可被使用的特定目录: public static String DIRECTORY_ALARMS = “Alarms”; public static String DIRECTORY_AUDIOBOOKS = “Audiobooks”; public static String DIRECTORY_DCIM = “DCIM”; public static String DIRECTORY_DOCUMENTS = “Documents”; public static String DIRECTORY_DOWNLOADS = “Download”; public static String DIRECTORY_MOVIES = “Movies”; public static String DIRECTORY_MUSIC = “Music”; public static String DIRECTORY_NOTIFICATIONS = “Notifications”; public static String DIRECTORY_PICTURES = “Pictures”; public static String DIRECTORY_PODCASTS = “Podcasts”;
第二步:使用FileProvider
在这之前,我们需要在AndroidManifest.xml中增加必要的读写权限:
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
1. 通过相机获取图片
在通过Intent跳转系统相机前,我们需要对版本进行判断,如果在Android7.0以上,使用FileProvider获取Uri,代码如下:
/**
* 从相机获取图片
*/
private void getImgFromCamera() {
//用于保存调用相机拍照后所生成的文件
tempFile = new File(this.getExternalFilesDir(DIRECTORY_DCIM), System.currentTimeMillis() + ".png");
//跳转到调用系统相机
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
//判断版本
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { //如果在Android7.0以上,使用FileProvider获取Uri
intent.setFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
Uri contentUri = FileProvider.getUriForFile(MainActivity.this, "com.choosecrop.fileprovider", tempFile);
intent.putExtra(MediaStore.EXTRA_OUTPUT, contentUri);
} else { //否则使用Uri.fromFile(file)方法获取Uri
intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(tempFile));
}
startActivityForResult(intent, CAMERA_REQUEST_CODE);
}
2.通过相册获取图片
/**
* 从相册获取图片
*/
private void getImgFromDicm() {
Intent photo = new Intent(Intent.ACTION_PICK);
photo.setType("image/*");
photo(photoPickerIntent, DICM_REQUEST_CODE);
}
3.剪裁图片
/**
* 裁剪图片
*/
private void cropPhoto(Uri uri) {
Intent intent = new Intent("com.android.camera.action.CROP");
// Intent intent = new Intent("android.intent.action.EDIT");
// intent.setAction("android.intent.action.EDIT");
// intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
intent.setDataAndType(uri, "image/*");
// intent.setFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
intent.putExtra("crop", "true");
intent.putExtra("aspectX", 1);
intent.putExtra("aspectY", 1);
intent.putExtra("outputX", 100);
intent.putExtra("outputY", 100);
intent.putExtra("return-data", false);
File cropTemp = this.getExternalFilesDir(DIRECTORY_DCIM);
File cropTempName = new File(cropTemp, System.currentTimeMillis() + "_crop_temp.png");
Log.e("getPath", cropTempName.getAbsolutePath());
Uri uriForFile = FileProvider.getUriForFile(this, "com.choosecrop.fileprovider", cropTempName);
intent.putExtra(MediaStore.EXTRA_OUTPUT, uriForFile);
grantPermissionFix(intent, uriForFile);
startActivityForResult(intent, CROP_REQUEST_CODE);
}
此处应设置return-data为false。 如果设置为true,是直接返回bitmap格式的数据,不仅耗费内存,最重要的是在Android 12,会报错: src:content://com.miui.gallery.open/raw/%2Fstorage%2Femulated%2F0%2FPictures%2FWeiXin%2Fmmexport1662718600424.jpg, des: null。
所以设置为false。然后,设置裁剪完之后保存的路径,在这里使用URI方式传递数据,即:intent.putExtra(MediaStore.EXTRA_OUTPUT, uriForFile); 同时需要给系统裁剪APP赋予临时权限,
private void grantPermissionFix(Intent intent, Uri uri) {
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
List<ResolveInfo> resolveInfos = getPackageManager().queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY);
for (ResolveInfo resolveInfo : resolveInfos) {
String packageName = resolveInfo.activityInfo.packageName;
grantUriPermission(packageName, uri, Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
intent.setAction(null);
intent.setComponent(new ComponentName(resolveInfo.activityInfo.packageName, resolveInfo.activityInfo.name));
break;
}
}
第三步:接收图片信息
在onActivityResult方法中获得返回的图片信息, 先调用剪裁去剪裁图片,然后对剪裁返回的图片进行设置、保存、上传等操作。
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent intent) {
switch (requestCode) {
case CAMERA_REQUEST_CODE: //调用相机后返回
if (resultCode == RESULT_OK) {
//用相机返回的照片去调用剪裁也需要对Uri进行处理
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
Uri contentUri = FileProvider.getUriForFile(MainActivity.this, "com.choosecrop.fileprovider", tempFile);
cropPhoto(contentUri);
} else {
cropPhoto(Uri.fromFile(tempFile));
}
}
break;
case DICM_REQUEST_CODE: //调用相册后返回
if (resultCode == RESULT_OK) {
Uri uri = intent.getData();
if (null != uri)
cropPhoto(uri);
else {
Log.e("e", "null");
}
}
break;
case CROP_REQUEST_CODE: //调用剪裁后返回
Log.e("d", "--------------222222222-------");
if (null != intent) {
Log.e("d", "---------------------not null");
Uri data = intent.getData();
mHeader_iv.setImageURI(data);
// Bundle bundle = intent.getExtras();
// if (bundle != null) {
// //在这里获得了剪裁后的Bitmap对象,可以用于上传
// Bitmap image = bundle.getParcelable("data");
// //设置到ImageView上
// mHeader_iv.setImageBitmap(image);
// //也可以进行一些保存、压缩等操作后上传
String path = saveImage("crop", image);
// }
} else {
Log.e("d", "---------null");
}
break;
}
}
因为使用uri方式传递的文件,所以这样获取数据, Uri data = intent.getData();
第四步:保存
public String save(String name, Bitmap bmp) {
File dir = new File(this.getExternalFilesDir(DIRECTORY_DCIM).getPath());
if (!dir.exists()) {
dir.mkdir();
}
String fileName = name + ".png";
File file = new File(dir, fileName);
try {
FileOutputStream fos = new FileOutputStream(file);
bmp.compress(Bitmap.CompressFormat.PNG, 100, fos);
fos.flush();
fos.close();
return file.getAbsolutePath();
} catch (IOException e) {
e.printStackTrace();
}
return null;
}
至此,Android 12 的兼容就OK了。
完整代码,可以到Github下载,别忘给star哦:
https://github.com/AlbertShen0211/ChooseCrop
|