最近深入了解了關于圖片的選擇和裁剪部分的內容,發現還是有很多需要注意的點,需要及時的記錄一下。
根據使用場景的不同,需要分情況討論。
先貼代碼再分析:
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()
函數里,如果保存圖片的目錄不存在,需要手動創建,否則無法保存裁剪后的圖片。