完全掌握 Android 选择图片、拍照以及图片裁剪
2016-05-19 本文已影响3890人
VitaminChen
最近深入了解了关于图片的选择和裁剪部分的内容,发现还是有很多需要注意的点,需要及时的记录一下。
根据使用场景的不同,需要分情况讨论。
先贴代码再分析:
Uri imageUri;
/* 场景1:选择一张图片 */
private void gotoPickImage() {
Intent intent = new Intent(Intent.ACTION_PICK, MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
startActivityForResult(intent, REQUEST_PICK_IMAGE);
}
/* 场景2:选择一张图片并裁剪获得一个小图 */
private void gotoPickAndCropSmallBitmap() {
Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
intent.setType("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("return-data", true);
intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
intent.putExtra("noFaceDetection", true); // no face detection
startActivityForResult(intent, REQUEST_CROP_IMAGE_SMALL);
}
/* 场景3:选择一张图片并裁剪获得一个大图 */
private void gotoPickAndCropBigBitmap() {
imageUri = getTmpUri();
Intent intent = new Intent(Intent.ACTION_GET_CONTENT, null);
intent.setType("image/*");
intent.putExtra("crop", "true");
intent.putExtra("aspectX", 1);
intent.putExtra("aspectY", 1);
intent.putExtra("outputX", 2000);
intent.putExtra("outputY", 2000);
intent.putExtra("scale", true);
intent.putExtra("return-data", false);
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
intent.putExtra("noFaceDetection", true); // no face detection
startActivityForResult(intent, REQUEST_CROP_IMAGE_BIG);
}
/* 场景4:拍照并裁剪 */
private void startImageCapture() {
String IMAGE_FILE_LOCATION = Environment.getExternalStorageDirectory() + "/" + "posprint" + "/tmp.jpg";//temp file
imageUri = Uri.fromFile(new File(IMAGE_FILE_LOCATION));//The Uri to store the big bitmap
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
startActivityForResult(intent, REQUEST_CAPTURE_AND_CROP);
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (resultCode == Activity.RESULT_OK) {
Bitmap bitmap = null;
try {
switch (requestCode) {
case REQUEST_PICK_IMAGE:
//选择的图片的Uri
imageUri = data.getData();
bitmap = MediaStore.Images.Media.getBitmap(
getContentResolver(), imageUri);
case REQUEST_CROP_IMAGE_SMALL:
//裁剪后的小图
bitmap = data.getParcelableExtra("data");
break;
case REQUEST_CROP_IMAGE_BIG:
//裁剪后的大图
bitmap = MediaStore.Images.Media.getBitmap(getContentResolver(), imageUri);
break;
case REQUEST_CAPTURE_AND_CROP:
//得到拍照后的图片并裁剪
cropImageUri(imageUri, REQUEST_CROP_IMAGE_BIG);
break;
}
} catch (IOException e) {
e.printStackTrace();
}
}
doSomething(bitmap);
}
//获得临时保存图片的Uri,用当前的毫秒值作为文件名
private Uri getTmpUri() {
String IMAGE_FILE_DIR = Environment.getExternalStorageDirectory() + "/" + "app_name";
File dir = new File(IMAGE_FILE_DIR);
File file = new File(IMAGE_FILE_DIR, Long.toString(System.currentTimeMillis()));
//非常重要!!!如果文件夹不存在必须先手动创建
if (!dir.exists()) {
dir.mkdirs();
}
return Uri.fromFile(file);
}
//裁剪拍照后得到的图片
private void cropImageUri(Uri uri, int requestCode) {
Intent intent = new Intent("com.android.camera.action.CROP");
intent.setDataAndType(uri, "image/*");
//intent.putExtra("crop", "true");
intent.putExtra("aspectX", 1);
intent.putExtra("aspectY", 1);
intent.putExtra("outputX", 500);
intent.putExtra("outputY", 500);
intent.putExtra("scale", true);
intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
intent.putExtra("return-data", false);
intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
intent.putExtra("noFaceDetection", true); // no face detection
intent = Intent.createChooser(intent, "裁剪图片");
startActivityForResult(intent, requestCode);
}
场景1:
直接从图库中选择一张照片,在 onActivityResult()
中就可以通过 data.getData()
得到该图片的Uri,没什么好说的
场景2:
从图库中选择一张照片并 裁剪得到一个小图。这里为什么强调是小图,这是因为当图片过大时(我在魅族 Pro5 上测试分辨率 300 * 300 时正常,当设为 400 * 400 就出错了,具体临界点不明)。
这里要注意几个 extra 字段,先上一个图:
裁剪图片的 extra 字段
-
return-data
: 设为true
的时候,在onActivityResult()
中可以直接通过data.getParcelableExtra("data")
得到裁剪后的 Bitmap 对象。但是当 Bitmap 过大时,就不能使用这种方法了,必须使用 场景3 中采用的方法。 -
MediaStore.EXTRA_OUTPUT
: 设置裁剪图片的输入 Uri。可以通过Uri.fromFile(tmpFile)
获得
场景3:
和场景2十分类似,唯一的不同只是此时可以裁剪得到一个分辨率较高的图片。区别只在于 return-data
和 MediaStore.EXTRA_OUTPUT
这两个 extra 的设置。
这里把return-data
设为 false
,同时向 MediaStore.EXTRA_OUTPUT
设置一个临时构造的 Uri,这个 Uri 就用来保存裁剪后的大图。裁剪之后,在 onActivityResult()
中就可以通过 MediaStore.Images.Media.getBitmap( getContentResolver(), imageUri)
得到裁剪后的 Bitmap 大图。
场景4:
拍照后并裁剪,其实是分了两步。第一步拍照得到图片的 Uri,第二部把该图片的 Uri 传给裁剪图片的程序处理。
几点需要注意的地方:
- 这里的的 action 变成了
com.android.camera.action.CROP
, 并且不再需要设置 extra 字段crop
。 -
intent.setDataAndType(uri, "image/*")
,这里的 uri 指向拍照后得到的原图,intent.putExtra(MediaStore.EXTRA_OUTPUT, uri)
裁剪后的图片也保存在这个 uri,所以原图就被覆盖了。如果想同时保存原图和裁剪后的图片,这里需要再提供另外一个 Uri 对象 - 最终得到裁剪后的图片的方法和场景3是一样的。当然如果只是想得到一个裁剪后的小图的话,那么也可以通过场景2的方法得到。
注意事项:
- 注意添加读写权限及拍照权限:
<uses-permission android:name="android.permission.CAMERA"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
- 注意
getTmpUri()
函数里,如果保存图片的目录不存在,需要手动创建,否则无法保存裁剪后的图片。