android 7.0 手机拍照裁剪问题

在android7.0 之前 ,如果需要用到拍照的时候 我们一般都这样去写

//先新建一个File类型的对象
File file=new File(Environment.getExternalStorageDirectory(), "/img/"+System.currentTimeMillis() + ".jpg");
if (!file.getParentFile().exists())file.getParentFile().mkdirs();
//然后获取到file对象的uri对象
Uri imageUri = Uri.fromFile(file);
Intent intent = new Intent();
//设置Action为拍照
intent.setAction(MediaStore.ACTION_IMAGE_CAPTURE);
//将拍取的照片保存到指定URI
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
startActivityForResult(intent,1);

但是在7.0以上 或者 小米等机型的手机时,会发现在onActivityResult中,data为空,甚至会报以下错误

android.os.FileUriExposedException: ** exposed beyond app through Intent.getData()

原因:

在Android7.0系统上,Android 框架强制执行了 StrictMode API 政策禁止向你的应用外公开 file:// URI。 如果一项包含文件 file:// URI类型 的 Intent 离开你的应用,应用失败,并出现 FileUriExposedException 异常,建议使用content 路径

解决方案:

https://developer.android.google.cn/training/secure-file-sharing/setup-sharing.html (官方文档)

首先在清单文件AndroidManifest.xml中添加provider节点

 <provider
       android:name="android.support.v4.content.FileProvider"
       android:authorities="包名.fileProvider"//fileProvider为 FileProvider.getUriForFile的方法名
       android:grantUriPermissions="true"//表示uri访问授权;
       android:exported="false">//报安全异常 true的时候  有可能会报安全异常
  <meta-data
       android:name="android.support.FILE_PROVIDER_PATHS"
       android:resource="@xml/file_path"/>//需要在xml文件夹中创建file_path文件
 </provider>

file_path.xml 文件如下

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <external-path name="external_files" path="."/>
</paths>

/**
*上述代码中 path=“.”,是有特殊意义的,它指的是根目录,也就是说您可以向其它的应用访问根目录及其子目
*录下任何一个文件了,如果您将path设为 path=”pictures”,那么它代表着根目录下的pictures目录
*(eg:/storage/emulated/0/pictures),这时您访问pictures目录范围之外的文件是不行的。
*files-path代表的根目录: Context.getFilesDir()
*external-path代表的根目录: Environment.getExternalStorageDirectory()
*cache-path代表的根目录: getCacheDir()
*/

拍照的代码:

//新建一个file对象
file = new File(Environment.getExternalStorageDirectory() + "/image.jpg");
//通过FileProvider.getUriForFile()方法获取到file的uri路径,"包名.fileProvider"要和清单文件
//中的android:authorities属性相对应
Uri imageUri = FileProvider.getUriForFile(getActivity(), "包名.fileProvider", file);
Intent intent = new Intent();
intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
// 设置Action为拍照
intent.setAction(MediaStore.ACTION_IMAGE_CAPTURE);
// 将拍取的照片保存到指定URI
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
// 开启一个带有返回值的Activity,请求码为TAKE_PICTURE
startActivityForResult(intent, TAKE_PICTURE);

按照上述代码写出来的拍照,在onActivityResult中得到的data为空,所以想对拍照结果进行处理,必须使用拍照之前新建的File对象

拍照剪切代码如下:

private void crop() {
		Uri uri;
		// 裁剪图片意图
		Intent intent = new Intent("com.android.camera.action.CROP");
		if (Build.VERSION.SDK_INT >= 24) {
			//添加这一句表示对目标应用临时授权该Uri所代表的文件
			intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
			//通过FileProvider创建一个content类型的Uri
			uri = FileProvider.getUriForFile(getActivity(), "包名.fileProvider", file);
		} else {
            //在API小于24的情况下 通过Uri.fromFile()获取到uri路径
			uri = Uri.fromFile(file);
		}
		intent.setDataAndType(uri, "image/*");
		intent.putExtra("crop", "true");
		// 裁剪框的比例,1:1
		intent.putExtra("aspectX", 1);
		intent.putExtra("aspectY", 1);
		// 裁剪后输出图片的尺寸大小
		intent.putExtra("outputX", 500);
		intent.putExtra("outputY", 500);
		// 取消人脸识别
		intent.putExtra("noFaceDetection", true);
		intent.putExtra("scale", true);
		intent.putExtra("return-data", false);
		intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.parse("file://" + "/" + Environment.getExternalStorageDirectory() + "/" + "image.jpg"));
		// 图片格式
		intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
		// 开启一个带有返回值的Activity,请求码为CROP_SMALL_PICTURE
		startActivityForResult(intent, CROP_SMALL_PICTURE);
	}

最后在请求码为CROP_SMALL_PICTURE 的时候,做最后的操作,比如图片的上传,展示等等

Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(uritempFile));
imageView.setImageBitmap(bitmap);

以上为拍照的完整流程,

当然,如果你不想对拍的照片进行裁剪就使用的话,那么最初创建的File就是你的拍照文件,是可以直接拿来使用的,请先判断非空哦!!!

==========================================分割线===============================================

从图库选择的代码如下

//同样创建一个file对象,只是为了使用方便
file = new File(Environment.getExternalStorageDirectory() + "/image.jpg");

Intent intentFromGallery = new Intent(Intent.ACTION_PICK, null);
//创建Intent意图				intentFromGallery.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, "image/*");
// 开启一个带有返回值的Activity,请求码为CHOOSE_PICTURE
startActivityForResult(intentFromGallery, CHOOSE_PICTURE);

使用图库选择照片之后,在onActivityResult中得到的data对象,通过data.getData()方法拿到选择图片的Uri路径,然后对图片进项裁剪,具体代码如下:

//判断data是否为空
if (data != null){
	crop(data.getData());
}
private void crop(Uri uri) {
		// 裁剪图片意图
		Intent intent = new Intent("com.android.camera.action.CROP");
		intent.setDataAndType(uri, "image/*");
		intent.putExtra("crop", "true");
		// 裁剪框的比例,1:1
		intent.putExtra("aspectX", 1);
		intent.putExtra("aspectY", 1);
		// 裁剪后输出图片的尺寸大小
		intent.putExtra("outputX", 500);
		intent.putExtra("outputY", 500);
		// 取消人脸识别
		intent.putExtra("noFaceDetection", true);
		intent.putExtra("scale", true);
		intent.putExtra("return-data", false);
		intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.parse("file://" + "/" + Environment.getExternalStorageDirectory() + "/" + "image.jpg"));
		// 图片格式
		intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
		// 开启一个带有返回值的Activity,请求码为CROP_SMALL_PICTURE
		startActivityForResult(intent, CROP_SMALL_PICTURE);
	}

最后在请求码为CROP_SMALL_PICTURE 的时候,做最后的操作,比如图片的上传,展示等等

 

最后附上 不需要对图库照片进行剪切使用方法:

首先需要在onActivityResult中得到的data对象,通过data.getData()方法拿到选择图片的Uri路径,然后把Uri路径转为文件的绝对路径,就不上代码了,具体是参考:

https://www.jianshu.com/p/b168cbe50066

然后就可以通过new 一个file对象使用了

 

最后大家前往不要忘了 权限申请哦!!!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值