Android 7.0 打开文件的错误,调用相机及裁剪图片,还有动态权限
- 华为荣耀V8,小米5,小米6,R9s,测试通过,其他机子没机会测试。。。。
核心代码
- 7.0以上除了相机和裁剪图片,只要是打开文件的,uri都要更改,包括更新安装包,记得要改uri,不然解析异常,但裁剪图片outputUri有点不一样
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { //赋予权限 intent.setFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); Uri uri = FileProvider.getUriForFile(activity, Constants.FILE_CONTENT_FILEPROVIDER, file); //举个栗子 intent.setDataAndType(uri,"application/" + fileType); } else { intent.setDataAndType(Uri.fromFile(file),"application/" + fileType); }
- 因为公司项目需要能修改头像功能,之前的测试机一直都是Android 5.1的机子,一直没问题,突然有一天,小米5升级到Android 7.0,华为机子也升级上去,然后就一直奔溃,然后就开始上网查资料啦。。。
其中相机调用正常通过,但是裁剪就不是了,这次重点就是裁剪EXTRA_OUTPUT,
别用 FileProvider.getUriForFile,否者在onActivityResult时,返回的resultCode为0,即取消。 - 首先Android6.0以上开始要加入动态权限管理,所以开启相机之前,动态权限要去申请,这里我就不多说了,网上很多,推荐一个大牛的:
转自严振杰的博客:http://blog.csdn.net/yanzhenjie1003/article/details/52503533
以及他的Github:https://github.com/yanzhenjie/AndPermission#userconsent# - 网上关于7.0的适配,我也是看到简书上有看到:http://www.jianshu.com/p/56b9fb319310
这个大家也能看看,我的就是参考他的。
上面2个建议大家都看看,很有用的,我的就是基于他们做的。
1.首先res文件夹下,新建一个xml文件夹,名字就是 android:resource="@xml/file_paths"对应的内容
<?xml version="1.0" encoding="utf-8"?> <resources> <paths> <external-path path="" name="camera_photos" /> </paths> </resources>
2.在Manifest <application>里面添加下面一段
<provider android:name="android.support.v4.content.FileProvider" android:authorities="com.goodbao.furniture.fileprovider" android:grantUriPermissions="true" android:exported="false"> <meta-data android:name="android.support.FILE_PROVIDER_PATHS" android:resource="@xml/file_paths" /> </provider>
android:authorities这个属性的值,我写的是包名+fileprovider,其实不是很重要,后面会用的到
其他属性的介绍,http://www.jianshu.com/p/56b9fb319310 这篇里面讲的听清楚的了。
3.调用系统相机
photo_image = createImagePath(APP_NAME + DATE); File file = new File(photo_image); if (!file.getParentFile().exists()) { file.getParentFile().mkdirs(); } Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); //Android7.0以上URI if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { //添加这一句表示对目标应用临时授权该Uri所代表的文件 intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); //通过FileProvider创建一个content类型的Uri Uri uri = FileProvider.getUriForFile(activity, Constants.FILE_CONTENT_FILEPROVIDER, file); intent.putExtra(MediaStore.EXTRA_OUTPUT, uri); } else { intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(file)); } try { activity.startActivityForResult(intent, REQUEST_CODE_TAKE_PHOTO); } catch (ActivityNotFoundException anf) { ToastUtils.showShortToast("摄像头尚未准备好")); }
其中
//添加这一句表示对目标应用临时授权该Uri所代表的文件 intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); //通过FileProvider创建一个content类型的Uri Uri uri = FileProvider.getUriForFile(activity, Constants.FILE_CONTENT_FILEPROVIDER, file); intent.putExtra(MediaStore.EXTRA_OUTPUT,uri);
- Constants.FILE_CONTENT_FILEPROVIDER这个值
就是之前 Manifest 里添加的 android:authorities="com.goodbao.furniture.fileprovider", - 所以你也能写成
Uri uri = FileProvider.getUriForFile(activity, "com.goodbao.furniture.fileprovider", file);
只是多次用到,我就弄成共用的。 - Android7.0以上,相机调用时,intent.putExtra(MediaStore.EXTRA_OUTPUT, uri),
Uri就不能用Uri.fromFile(file)
而是要FileProvider.getUriForFile(activity, Constants.FILE_CONTENT_FILEPROVIDER, file);
但是裁剪的时候就不一样,intent.setData,即打开图片的Uri需要使用ContentUri,
而intent.putExtra(MediaStore.EXTRA_OUTPUT, outputUri);里的outputUri继续使用 Uri.fromFile(file),
估计是裁剪操作一整个流程,应该是授权一次就好,在访问时先授权,而写入就不需要了。
4.裁剪图片
/** * 调用系统剪裁功能 */ public void cropPicture(Activity activity, String path) { File file = new File(path); if (!file.getParentFile().exists()) { file.getParentFile().mkdirs(); } Uri imageUri; Uri outputUri; crop_image = createImagePath(APP_NAME + "_crop_" + DATE); Intent intent = new Intent("com.android.camera.action.CROP"); if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { //添加这一句表示对目标应用临时授权该Uri所代表的文件 intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); //TODO:访问相册需要被限制,需要通过FileProvider创建一个content类型的Uri imageUri = FileProvider.getUriForFile(activity, FILE_CONTENT_FILEPROVIDER, file); outputUri = Uri.fromFile(new File(crop_image)); //TODO:裁剪整个流程,估计授权一次就好outputUri不需要ContentUri,否则失败 //outputUri = FileProvider.getUriForFile(activity, "com.solux.furniture.fileprovider", new File(crop_image)); } else { imageUri = Uri.fromFile(file); outputUri = Uri.fromFile(new File(crop_image)); } intent.setDataAndType(imageUri, "image/*"); intent.putExtra("crop", "true"); //设置宽高比例 intent.putExtra("aspectX", 1); intent.putExtra("aspectY", 1); //设置裁剪图片宽高 intent.putExtra("outputX", 300); intent.putExtra("outputY", 300); intent.putExtra("scale", true); intent.putExtra(MediaStore.EXTRA_OUTPUT, outputUri); intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString()); intent.putExtra("noFaceDetection", true); activity.startActivityForResult(intent, REQUEST_CODE_CROP_PICTURE); }
- 要处理的图片imageUri , intent.setDataAndType(imageUri, "image/*");
imageUri = FileProvider.getUriForFile(activity, Constants.FILE_CONTENT_FILEPROVIDER, file); - 处理完的图片outputUri ,intent.putExtra(MediaStore.EXTRA_OUTPUT, outputUri);
outputUri = Uri.fromFile(new File(crop_image));就不需要,切记。