Android 拍照和相冊選圖 版本兼容

Android 拍照和相冊選圖 版本兼容


一.相機部分


    1. 調用啓動系統相機

       

intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

startActivityForResult(intent, CAMERA_REQUEST_CODE);


       常量值可以自己定義(private static int CAMERA_REQUEST_CODE = 1)


    2.Activity中方法回調

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (requestCode == CAMERA_REQUEST_CODE) {
if (data == null) {
return;
} else { 
Bundle extras = data.getExtras();
if (extras != null) {
Bitmap bm = extras.getParcelable("data");
Uri uri = saveBitmap(bm);
startImageZoom(uri);
}
}
}
}


     其中saveBitmap()方法如下:

private Uri saveBitmap(Bitmap bm) {

File tmpDir = new File(Environment.getExternalStorageDirectory()
+ "/test");
if (!tmpDir.exists()) {
tmpDir.mkdir();
}
File img = new File(tmpDir.getAbsolutePath() + "test.png");
try {
FileOutputStream fos = new FileOutputStream(img);
bm.compress(Bitmap.CompressFormat.PNG, 85, fos);
fos.flush();
fos.close();
Toast.makeText(MainActivity.this, "Success", Toast.LENGTH_SHORT).show();
return Uri.fromFile(img);
} catch (FileNotFoundException e) {
Toast.makeText(MainActivity.this, "Failed", Toast.LENGTH_SHORT).show();
e.printStackTrace();
return null;
} catch (IOException e) {
e.printStackTrace();
Toast.makeText(MainActivity.this, "Failed", Toast.LENGTH_SHORT).show();
return null;
}


}


    startImageZoom()方法如下:


private void startImageZoom(Uri uri) {
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", 150);
intent.putExtra("outputY", 150);
intent.putExtra("return-data", true);
startActivityForResult(intent, CROP_REQUEST_CODE);
}


      其中輸出比例和輸出尺寸可以根據需要修改,常量值可以自己定義(private static int CROP_REQUEST_CODE = 3


    3.最後當然是接受裁剪後的圖片,並顯示給控件了


@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (requestCode == CROP_REQUEST_CODE) {
if (data == null) {
return;
}else{
Bundle extras = data.getExtras();
if (extras == null) {
return;
}
Bitmap bm = extras.getParcelable("data");

//TODO

這裏當然是顯示Bitmap 給圖片控件了

}

}
}



二.相冊部分

     1.調用啓動系統相冊

intent = new Intent(Intent.ACTION_GET_CONTENT);

intent.setType("image/*");

startActivityForResult(intent, GALLERY_REQUEST_CODE);


      常量可以自己定義(private static int GALLERY_REQUEST_CODE = 2)


    2.Activity回調

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (requestCode == GALLERY_REQUEST_CODE) {
if (data == null) {
return;
}else{
Uri uri = data.getData();
Uri fileUri = convertUri(uri);
startImageZoom(fileUri);
}


}

       其中convertUri()方法如下:

private Uri convertUri(Uri uri) {
InputStream is = null;
try {
is = getContentResolver().openInputStream(uri);
Bitmap bitmap = BitmapFactory.decodeStream(is);
is.close();
return saveBitmap(bitmap);
} catch (FileNotFoundException e) {
e.printStackTrace();
return null;
} catch (IOException e) {
e.printStackTrace();
return null;
}
}


       其中startImageZoom()方法,上文已寫


      3.獲取裁剪後圖片,並設置給圖片控件


        與相機獲取裁剪圖片是一個方法


三. 請記得添加權限啊,很重要

     

<uses-permission android:name="android.permission.CAMERA" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>


It's Over,下邊介紹2中兼容性問題解決方案(別人總結的,借過來用用)

     1.選擇系統相冊時HTC 7出現的系統崩潰(空指針異常) 系統版本 2.3.7

        通常從系統相冊中選擇圖片寫法如下:

albumButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(Intent.ACTION_PICK, android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
startActivityForResult(intent, AppContext.GETIMAGE_BYSDCARD);
}
});

      然後在onActivityResult中獲取剛剛選取的照片:

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent intent) {
if (resultCode == RESULT_OK) {
if (requestCode == AppContext.GETIMAGE_BYSDCARD || requestCode == AppContext.GETIMAGE_BYCAMERA) {
if (requestCode == AppContext.GETIMAGE_BYSDCARD && null != data) {
Uri selectedImage = data.getData();
String[] filePathColumn = { MediaStore.Images.Media.DATA };
Cursor cursor = getContentResolver().query(selectedImage, filePathColumn, null, null, null);
 if (cursor != null) {
 if (cursor.moveToFirst()) {
//int columnIndex = cursor.getColumnIndex(filePathColumn[0]);
 int columnIndex = cursor.getColumnIndexOrThrow(filePathColumn[0]);
 photoPath = cursor.getString(columnIndex);
}
 cursor.close();
}
}
}
}
}

   可以在HTC7  2.3.7 上發現無法獲取圖片,如果上述代碼中沒有做cursor != null 則系統崩潰,最後定爲出原因在於Uri selectedImage = data.getData();這行代碼上,在其他手     機上,此處返回格式爲content://media/external/images/media/244709,因此自然是通過接下來的Content Privider方式獲取到圖片實際地址。而在HTC此手機上,返回的結     果卻爲:/storage/sdcard0/DCIM/Camera/IMG_20140608_162447.jpg,即直接返回了所選取圖片的地址,因此,需要針對性的做出如下處理:


@Override
protected void onActivityResult(int requestCode, int resultCode, Intent intent) {
if (resultCode == RESULT_OK) {
if (requestCode == AppContext.GETIMAGE_BYSDCARD || requestCode == AppContext.GETIMAGE_BYCAMERA) {
if (requestCode == AppContext.GETIMAGE_BYSDCARD && null != data) {
Uri selectedImage = data.getData();
String[] filePathColumn = { MediaStore.Images.Media.DATA };
Cursor cursor = getContentResolver().query(selectedImage, filePathColumn, null, null, null);
if (cursor != null) {
 if (cursor.moveToFirst()) {
//int columnIndex = cursor.getColumnIndex(filePathColumn[0]);
 int columnIndex = cursor.getColumnIndexOrThrow(filePathColumn[0]);
 photoPath = cursor.getString(columnIndex);
}
cursor.close();
} else {
if (selectedImage != null) {
String tmpPath = selectedImage.getPath();
if (tmpPath != null && (tmpPath.endsWith(".jpg") || tmpPath.endsWith(".png") || tmpPath.endsWith(".gif"))) {
photoPath = tmpPath;
}
}
}
}
}
}
}

 2.三星手機調用手機拍照後出現橫豎屏切換的問題

   其實這也是一個典型的問題了,當初也是在網上直接查到的解決方案,貌似是三星手機的通病。解決方案如下:

public static int getPictureDegree(String path) {
int degree = 0;
try {
ExifInterface exifInterface = new ExifInterface(path);
int orientation = exifInterface.getAttributeInt(ExifInterface.TAG_ORIENTATION, ExifInterface.ORIENTATION_NORMAL);
switch (orientation) {
case ExifInterface.ORIENTATION_ROTATE_90:
degree = 90;
break;
case ExifInterface.ORIENTATION_ROTATE_180:
degree = 180;
break;
case ExifInterface.ORIENTATION_ROTATE_270:
degree = 270;
break;
}
} catch (IOException e) {
e.printStackTrace();
}
return degree;
}


int degree = getPictureDegree(filePath);
return roateBitmap(thumbBitmap, degree);


public static Bitmap roateBitmap(Bitmap bitmap, int degree) {
if (degree == 0) {
return bitmap;
}
Matrix matrix = new Matrix();
matrix.postRotate(degree);
Bitmap bmp = Bitmap.createBitmap(bitmap, 0, 0, bitmap.getWidth(), bitmap.getHeight(), matrix, true);
return bmp;
}


主要思路其實就是通過圖片Exif獲取到其旋轉角度,然後再相應旋轉過來。


源碼地址:https://github.com/FreetoflyBai/CameraManager

 

發佈了12 篇原創文章 · 獲贊 10 · 訪問量 7萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章