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));就不需要,切记。