拍照獲取圖片流程
- 創建一個File對象,用於裝相機拍下的照片,起名:output_image.jpg
- 放到當前應用關聯緩存目錄下,getCacheDir()/getExternalCacheDir()
- 以android 7.0 爲界,用不同的方式將File轉換成Uri對象
- 然後構建Intent ,傳入Uri,隱式啓動相機
- onActivityResult()得到返回值並處理
創建File對象
先創建一個file
//創建File對象,用於存儲拍照後的圖片
File outputImage = new File(getDiskCacheDir(this), "output_img.jpg");
//判斷該目錄下是否有文件存在
try {
if (outputImage.exists()) {
outputImage.delete();
}
outputImage.createNewFile();
} catch (IOException e) {
e.printStackTrace();
}
放到當前應用關聯緩存目錄
獲取可以使用的緩存地址:這裏以sd卡的緩存目錄爲優先,sd卡不能用就用手機本地緩存目錄。
- 應用關聯緩存目錄:SD卡中專門存放當前應用緩存數據的位置;
調用getExternalCacheDir()方法可以得到這個目錄,在卸載程序的時候,這裏數據可以被一起刪除,而且用緩存目錄的話即便是6.0以上的系統也不需要動態獲取讀寫權限,因爲不在內存而是緩存
/**
* 獲取緩存地址,如果sd卡不存在就獲取手機應用關聯緩存目錄
* 卸載程序可被刪除,android6.0後不需要請求讀寫權限
* getExternalCacheDir():/sdcard/Android/data/<application package>/cache
* getCacheDir():/data/data/<application package>/cache
* @param context
* @return
*/
public String getDiskCacheDir(Context context) {
String cachePath = null;
//Environment.getExtemalStorageState() 獲取SDcard的狀態
//Environment.MEDIA_MOUNTED 只有改狀態下纔可以進行讀寫
if (Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState())
|| !Environment.isExternalStorageRemovable()) {
//sd卡應用關聯緩存目錄
cachePath = context.getExternalCacheDir().getPath();
} else {
//手機應用關聯緩存目錄
cachePath = context.getCacheDir().getPath();
}
return cachePath;
}
將File轉換成Uri對象
啓動相機需要傳入有效的Uri才行,現在還是File格式,需要轉換。
- Android7.0 開始,直接使用本地真實路徑的Uri被認爲是不安全的,會拋出FileUriException異常。而FileProvider則是一種特殊的內容提供器,它使用了和內容提供器類似的機制來對數據進行保護。
//Android 7.0 開始,直接使用本地真實路徑的Uri被認爲是不安全的,而FileProvide則是一種特殊的內容提供器
if (Build.VERSION.SDK_INT >= 24) {
imgUri = FileProvider.getUriForFile(MainActivity.this, "com.yuanli.choosephoto.fileprovider", outputImage);
} else {
imgUri = Uri.fromFile(outputImage);
}
構建Intent ,傳入Uri,隱式啓動相機
//隱式啓動相機程序
Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");
intent.putExtra(MediaStore.EXTRA_OUTPUT, imgUri);
startActivityForResult(intent, TAKE_PHOTO);
onActivityResult()得到返回值並處理
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (resultCode == Activity.RESULT_CANCELED) {
return;
}
switch (requestCode) {
case TAKE_PHOTO:
if (resultCode == RESULT_OK) {
//將拍攝的照片展示出來
try {
Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imgUri));
imgChoose.setImageBitmap(bitmap);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
}
break;
default:
break;
}
}
寫着寫着跑偏了,研究相冊的時候需要動態申請權限,就想封裝一個好點的,結果看各種文檔一天就有過去了。還好差不多了。
被註解難住了,好難啊!看了各種文檔了,想自己寫不想用hy大神的庫,可仿不出hy大神的註解,好狗的難,到底哪裏沒想明白???
過了個週末。。。。。那麼久,,,,,繼續寫。。。
對不起,自己勉強明白了個大概,粗略的寫在了下一篇博客裏。不值得關注,有需要的還是自行查閱比較好。等我真正消化了,再好好真理一篇清晰易懂的。
繼續相機
相冊選取圖片流程
- 獲取讀寫權限
- 打開圖庫
- 處理返回結果
打開圖庫
獲取權限有很多方式,自己選擇喜歡的,我就直接說正題了
private void choosePhoto() {
//隱式啓動相冊
Intent intent = new Intent("android.intent.action.GET_CONTENT");
intent.setType("image/*");
startActivityForResult(intent, CHOOSE_PHOTO);
}
處理返回結果
啓動Intent時傳入數據是Uri格式,而Intent返回結果也是Uri格式的。只不過老規矩,4.4版本(19)爲界,4.4以上返回的Uri不是圖片真實的Uri,而是的封裝過的。這裏挺奇怪哈,拍照的時候給Intent傳入Uri的時候,界限是7.0(24);不懂咋想的。反正記住別混爲一談了。
返回值判斷:
case CHOOSE_PHOTO:
if(resultCode==RESULT_OK){
//判斷手機系統版本號
if(Build.VERSION.SDK_INT>=Build.VERSION_CODES.KITKAT){
//4.4ji以上系統使用這個方法處理圖片
handlerImageOnKitKat(data);
}else{
//4.4ji以下系統使用這個方法處理圖片
handlerImageBeforeKitKat(data);
}
}
break;
/**
* android系統從4.4版本開始,選取相冊中圖片不再返回圖片的真實的Uri了,
* 而是一個封裝過的Uri,因此需要對這個Uri進行解析才行。
* @param data
*/
@RequiresApi(api = Build.VERSION_CODES.KITKAT)
private void handlerImageOnKitKat(Intent data) {
String imagePath=null;
Uri uri=data.getData();
imagePath = uriToPath(uri);
// startPhotoZoom(new File(imagePath), 350);
displayImage(imagePath);
}
private String uriToPath(Uri uri) {
String path = null;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
if (DocumentsContract.isDocumentUri(this, uri)) {
// 如果是document類型的Uri,則通過document id處理
String docId = DocumentsContract.getDocumentId(uri);
//如果authority是media格式的話,document id 還需要在進行一次解析
if ("com.android.providers.media.documents".equals(uri.getAuthority())) {
String id = docId.split(":")[1]; // 解析出真正的數字格式的id
String selection = MediaStore.Images.Media._ID + "=" + id;
path = getImagePath(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, selection);
} else if ("com.android.providers.downloads.documents".equals(uri.getAuthority())) {
Uri contentUri = ContentUris.withAppendedId(Uri.parse("content://downloads/public_downloads"), Long.valueOf(docId));
path = getImagePath(contentUri, null);
}
} else if ("content".equalsIgnoreCase(uri.getScheme())) {
// 如果是content類型的Uri,則使用普通方式處理
path = getImagePath(uri, null);
} else if ("file".equalsIgnoreCase(uri.getScheme())) {
// 如果是file類型的Uri,直接獲取圖片路徑即可
path = uri.getPath();
}
}
return path;
}
private String getImagePath(Uri uri, String selection) {
String path = null;
// 通過Uri和selection來獲取真實的圖片路徑
Cursor cursor = getContentResolver().query(uri, null, selection, null, null);
if (cursor != null) {
if (cursor.moveToFirst()) {
path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA));
}
cursor.close();
}
return path;
}
/**
* android系統從4.4版本一下執行
* @param data
*/
private void handlerImageBeforeKitKat(Intent data) {
Uri uri = data.getData();
String imagePath = getImagePath(uri, null);
displayImage(imagePath);
Log.i("TAG", "file://" + imagePath + "選擇圖片的URI" + uri);
// startPhotoZoom(new File(imagePath), 350);
}
private void displayImage(String imagePath) {
if(imagePath!=null){
Bitmap bitmap=BitmapFactory.decodeFile(imagePath);
imgChoose.setImageBitmap(bitmap);
}
}