Android 4.4前後のバージョンでライブラリ画像を読み取り、写真を撮る完璧なソリューション
23519 ワード
転載:http://blog.csdn.net/zbjdsbj/article/details/42387551
4.3以下、画像を选んだ后に、Uriによって処理をして、多くの招待状はすべてあって、私は详しく言いません.主に4.4で、もし上のpickの原生の方法を使って図を选ぶならば、戻るuriはやはり正常で、しかしACTION_GET_CONTENTのメソッドでは、戻るuriは4.3とは全く異なり、4.3はファイルパス付きで、4.4はcontent://com.android.providers.media.documents/document/image:3951このように、経路がなく、ピクチャ番号のuriのみがあるため、ピクチャ経路に従ってトリミングすることができなくなる.
overflow大牛からのメソッド,4.4で得られたuriは,ファイルのパスを取得するために以下の方法が必要である.
どうして違うの?
Android 4.4(含む)からスタートし、方式でライブラリにアクセスした後、戻ってきたUriは以下のようになっています(「最近」).
それだけでなく、異なるタイプのライブラリでは、返されるUri形式は異なります(通常のアルバムにアクセス):
4.4以前に返されたUriには、次のような形式しか存在しません.
したがって、Android 4.4以降のデバイスでは、簡単なgetDataColumn(Context,Uri,null,null)によるピクチャデータベースはすべてのニーズを満たすことができないため、ピクチャの実際のパスを取得する際には、異なるタイプによって区別する必要がある.バージョン判定:
アルバムから写真の選び方比較:
表示などの処理は以下の通りです.
GIthubエンジニアリングアドレス:https://github.com/ZBJDSBJ/CameraDemo
4.3以下、画像を选んだ后に、Uriによって処理をして、多くの招待状はすべてあって、私は详しく言いません.主に4.4で、もし上のpickの原生の方法を使って図を选ぶならば、戻るuriはやはり正常で、しかしACTION_GET_CONTENTのメソッドでは、戻るuriは4.3とは全く異なり、4.3はファイルパス付きで、4.4はcontent://com.android.providers.media.documents/document/image:3951このように、経路がなく、ピクチャ番号のuriのみがあるため、ピクチャ経路に従ってトリミングすることができなくなる.
overflow大牛からのメソッド,4.4で得られたuriは,ファイルのパスを取得するために以下の方法が必要である.
/**
* <br> :4.4
* <br> :
* <br> :
* @param context
* @param uri
* @return
*/
@TargetApi(Build.VERSION_CODES.KITKAT)
public static String getPath(final Context context, final Uri uri) {
final boolean isKitKat = Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT;
// DocumentProvider
if (isKitKat && DocumentsContract.isDocumentUri(context, uri)) {
// ExternalStorageProvider
if (isExternalStorageDocument(uri)) {
final String docId = DocumentsContract.getDocumentId(uri);
final String[] split = docId.split(":");
final String type = split[0];
if ("primary".equalsIgnoreCase(type)) {
return Environment.getExternalStorageDirectory() + "/" + split[1];
}
}
// DownloadsProvider
else if (isDownloadsDocument(uri)) {
final String id = DocumentsContract.getDocumentId(uri);
final Uri contentUri = ContentUris.withAppendedId(
Uri.parse("content://downloads/public_downloads"), Long.valueOf(id));
return getDataColumn(context, contentUri, null, null);
}
// MediaProvider
else if (isMediaDocument(uri)) {
final String docId = DocumentsContract.getDocumentId(uri);
final String[] split = docId.split(":");
final String type = split[0];
Uri contentUri = null;
if ("image".equals(type)) {
contentUri = MediaStore.Images.Media.EXTERNAL_CONTENT_URI;
} else if ("video".equals(type)) {
contentUri = MediaStore.Video.Media.EXTERNAL_CONTENT_URI;
} else if ("audio".equals(type)) {
contentUri = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
}
final String selection = "_id=?";
final String[] selectionArgs = new String[] { split[1] };
return getDataColumn(context, contentUri, selection, selectionArgs);
}
}
// MediaStore (and general)
else if ("content".equalsIgnoreCase(uri.getScheme())) {
// Return the remote address
if (isGooglePhotosUri(uri))
return uri.getLastPathSegment();
return getDataColumn(context, uri, null, null);
}
// File
else if ("file".equalsIgnoreCase(uri.getScheme())) {
return uri.getPath();
}
return null;
}
public static String getDataColumn(Context context, Uri uri, String selection,
String[] selectionArgs) {
Cursor cursor = null;
final String column = "_data";
final String[] projection = { column };
try {
cursor = context.getContentResolver().query(uri, projection, selection, selectionArgs,
null);
if (cursor != null && cursor.moveToFirst()) {
final int index = cursor.getColumnIndexOrThrow(column);
return cursor.getString(index);
}
} finally {
if (cursor != null)
cursor.close();
}
return null;
}
/**
* @param uri The Uri to check.
* @return Whether the Uri authority is ExternalStorageProvider.
*/
public static boolean isExternalStorageDocument(Uri uri) {
return "com.android.externalstorage.documents".equals(uri.getAuthority());
}
/**
* @param uri The Uri to check.
* @return Whether the Uri authority is DownloadsProvider.
*/
public static boolean isDownloadsDocument(Uri uri) {
return "com.android.providers.downloads.documents".equals(uri.getAuthority());
}
/**
* @param uri The Uri to check.
* @return Whether the Uri authority is MediaProvider.
*/
public static boolean isMediaDocument(Uri uri) {
return "com.android.providers.media.documents".equals(uri.getAuthority());
}
/**
* @param uri The Uri to check.
* @return Whether the Uri authority is Google Photos.
*/
public static boolean isGooglePhotosUri(Uri uri) {
return "com.google.android.apps.photos.content".equals(uri.getAuthority());
}</span></span>
どうして違うの?
Android 4.4(含む)からスタートし、方式でライブラリにアクセスした後、戻ってきたUriは以下のようになっています(「最近」).
Uri is:content://com.android.providers.media.documents/document/image%3A18838
2 Uri.getPath is :/document/image:18838
3 :/storage/emulated/0/Pictures/Screenshots/Screenshot_2014-09-22-21-40-53.png</span>
それだけでなく、異なるタイプのライブラリでは、返されるUri形式は異なります(通常のアルバムにアクセス):
Uri is:content://media/external/images/media/18822
2 Uri.getPath is :/external/images/media/18822
3 :/storage/emulated/0/Download/20130224235013.jpg</span>
4.4以前に返されたUriには、次のような形式しか存在しません.
Uri is:content://media/external/images/media/14046
2 Uri.getPath is :/external/images/media/14046
3 :/storage/emulated/0/DCIM/Camera/20130224235013.jpg
したがって、Android 4.4以降のデバイスでは、簡単なgetDataColumn(Context,Uri,null,null)によるピクチャデータベースはすべてのニーズを満たすことができないため、ピクチャの実際のパスを取得する際には、異なるタイプによって区別する必要がある.バージョン判定:
// : 4.4
final boolean mIsKitKat = Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT;
アルバムから写真の選び方比較:
/**
* <br> :4.4
* <br> :
* <br> :
*/
@TargetApi(Build.VERSION_CODES.KITKAT)
private void SelectImageUriAfterKikat() {
Intent intent = new Intent(Intent.ACTION_OPEN_DOCUMENT);
intent.addCategory(Intent.CATEGORY_OPENABLE);
intent.setType("image/*");
startActivityForResult(intent, SELECET_A_PICTURE_AFTER_KIKAT);
}
/**
* <br> :4.4
* <br> :
* <br> :
*/
private void cropImageUri() {
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", 640);
intent.putExtra("outputY", 640);
intent.putExtra("scale", true);
intent.putExtra("return-data", false);
intent.putExtra(MediaStore.EXTRA_OUTPUT,
Uri.fromFile(new File(IMGPATH, TMP_IMAGE_FILE_NAME)));
intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
intent.putExtra("noFaceDetection", true); // no face detection
startActivityForResult(intent, SELECT_A_PICTURE);
}
, :
/**
* <br> :
* <br> :
* <br> :
* @param uri
*/
private void cameraCropImageUri(Uri uri) {
Intent intent = new Intent("com.android.camera.action.CROP");
intent.setDataAndType(uri, "image/jpeg");
intent.putExtra("crop", "true");
intent.putExtra("aspectX", 1);
intent.putExtra("aspectY", 1);
intent.putExtra("outputX", 640);
intent.putExtra("outputY", 640);
intent.putExtra("scale", true);
if (mIsKitKat) {
intent.putExtra("return-data", true);
} else {
intent.putExtra("return-data", false);
intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
}
intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
intent.putExtra("noFaceDetection", true);
startActivityForResult(intent, SET_PICTURE);
}
表示などの処理は以下の通りです.
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == SELECT_A_PICTURE) {
if (resultCode == RESULT_OK && null != data) {
Log.i("zou", "4.4 ");
Bitmap bitmap = decodeUriAsBitmap(Uri.fromFile(new File(IMGPATH,
TMP_IMAGE_FILE_NAME)));
mAcountHeadIcon.setImageBitmap(bitmap);
}
} else if (requestCode == SELECET_A_PICTURE_AFTER_KIKAT) {
if (resultCode == RESULT_OK && null != data) {
Log.i("zou", "4.4 ");
mAlbumPicturePath = getPath(MainActivity.this, data.getData());
cropImageUriAfterKikat(Uri.fromFile(new File(mAlbumPicturePath)));
}
} else if (requestCode == SET_ALBUM_PICTURE_KITKAT) {
Log.i("zou", "4.4 RESULT_OK");
Bitmap bitmap = data.getParcelableExtra("data");
mAcountHeadIcon.setImageBitmap(bitmap);
} else if (requestCode == TAKE_A_PICTURE) {
Log.i("zou", "resultCode:" + resultCode);
cameraCropImageUri(Uri.fromFile(new File(IMGPATH, IMAGE_FILE_NAME)));
} else if (requestCode == SET_PICTURE) {
Log.i("zou", "SET_PICTURE-resultCode:" + resultCode);
Bitmap bitmap = null;
if (mIsKitKat) {
if (null != data) {
bitmap = data.getParcelableExtra("data");
}
} else {
bitmap = decodeUriAsBitmap(Uri.fromFile(new File(IMGPATH, IMAGE_FILE_NAME)));
}
mAcountHeadIcon.setImageBitmap(bitmap);
}
}
GIthubエンジニアリングアドレス:https://github.com/ZBJDSBJ/CameraDemo