Android 7.0调用相机及裁剪图片,还有动态权限
简书:Android 7.0 文件读取适配,及适配相机及裁剪图片
Demo挂载在码云上 码云链接
Github代码 华为荣耀V8和小米5测试通过,其他机子没机会测试。。。。
因为公司项目需要能修改头像功能,之前的测试机一直都是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.首先在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 这篇里面讲的听清楚的了。
2.res文件夹下,新建一个xml文件夹,名字就是 android:resource="@xml/file_paths"对应的内容
<?xml version="1.0" encoding="utf-8"?> <resources> <paths> <external-path path="" name="camera_photos" /> </paths> </resources>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); 但是裁剪的时候就不一样,裁剪继续使用 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); //通过FileProvider创建一个content类型的Uri imageUri = FileProvider.getUriForFile(activity, Constants.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));就不需要,切记。