android相冊(cè)選擇圖片的編碼實(shí)現(xiàn)代碼
android相機(jī)拍照直接選取圖片固然方便,但是更多的時(shí)候,我們需要從手機(jī)已有的圖片中選擇一張來使用。這次就練習(xí)如何從相冊(cè)中選擇圖片吧。
首先在 activity_main.xml 文件中增加一個(gè) Button,用來觸發(fā)從相冊(cè)中選擇圖片的功能。
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:orientation="vertical" android:layout_width="match_parent" android:layout_height="match_parent" tools:context="com.cofox.mycameraalbum.MainActivity"> <Button android:id="@+id/button_take_photo" android:layout_width="match_parent" android:layout_height="wrap_content" android:textAllCaps="false" android:text="Tack Photo"/> <Button android:id="@+id/button_choose_from_album" android:layout_width="match_parent" android:layout_height="wrap_content" android:textAllCaps="false" android:text="Choose From Album"/> <ImageView android:id="@+id/photo_pictrue" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_gravity="center_horizontal"/> </LinearLayout>
這段代碼中的 android:id="@+id/button_choose_from_album" 按鈕就是我們的主角了。
然后要在MainActivity.java中,加入從相冊(cè)選圖片的邏輯。
聲明 Button 按鈕
Button btnChooseFromAlbum = (Button)findViewById(R.id.button_choose_from_album);//相冊(cè)選擇圖片按鈕
添加按鈕的點(diǎn)擊事件
//打開相冊(cè)選擇圖片 btnChooseFromAlbum.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { if(ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED){ ActivityCompat.requestPermissions(MainActivity.this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1); }else{ openAlbum(); } } });
這段代碼中,先要判斷 PackageManager.PERMISSION_GRANTED 權(quán)限的情況,如果沒有權(quán)限就需要添加,有的話直接 openAlbum(); 打開相冊(cè)。
ActivityCompat.requestPermissions 執(zhí)行獲取權(quán)限的操作,然后由用戶選擇是否給予權(quán)限,根據(jù)用戶的選擇,系統(tǒng)來決定是否可以繼續(xù)執(zhí)行 openAlbum()功能。這個(gè)就需要用到 onRequestPermissionsResult 了。代碼的邏輯是,如果給了權(quán)限就執(zhí)行 openAlbum(),如果不給權(quán)限就返回給用戶沒有此權(quán)限的提示即可。("You denied the permision.")
@Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { // super.onRequestPermissionsResult(requestCode, permissions, grantResults); switch (requestCode){ case 1: if(grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED){ openAlbum(); }else{ Toast.makeText(this, "You denied the permision.", Toast.LENGTH_LONG).show(); } break; default: } }
openAlbum() 就是打開相冊(cè),代碼邏輯很清晰吧?但是,還沒完呢。因?yàn)槿绾未蜷_還需要一些邏輯。先看 openAlbum() 的代碼。
private void openAlbum() { Intent intent = new Intent("android.intent.action.GET_CONTENT"); intent.setType("image/*"); startActivityForResult(intent, CHOOSE_PHOTO);//打開相冊(cè) }
這里的 Intent 是給相冊(cè)準(zhǔn)備的,然后調(diào)用 startActivityForResult() 才打開相冊(cè)。這時(shí)出現(xiàn)的 CHOOSE_PHOTO 是給 onActivityResult 中判斷走那個(gè)分支的條件參數(shù)。我們應(yīng)該在 MainActivity.java 中增加一個(gè)全局變量聲明
public static final int CHOOSE_PHOTO = 2;
在 onActivityResult 中增加switch case的 CHOOSE_PHOTO 分支。分之內(nèi)根據(jù)不同的系統(tǒng)版本執(zhí)行不同的代碼邏輯。因?yàn)?android 4.4 是一個(gè)文件訪問安全處理方式的分水嶺,4.4以下的系統(tǒng)使用直接文件地址,4.4 及以上系統(tǒng)使用不再返回真實(shí)的圖片地址了。所以,代碼的處理方法就有所不同。4.4及以上系統(tǒng)需要對(duì)相冊(cè)返回的圖片地址進(jìn)行解析。這里確定分作兩個(gè)方法來分別處理 handleImageOnKitKat(data) 和 handeleImageBeforeKitKat(data)。
case CHOOSE_PHOTO: if(resultCode == RESULT_OK){ //判斷手機(jī)系統(tǒng)版本號(hào) if(Build.VERSION.SDK_INT >= 19){ //4.4及以上系統(tǒng)使用這個(gè)方法處理圖片 handleImageOnKitKat(data); }else{ //4.4以下系統(tǒng)使用這個(gè)方法處理圖片 handeleImageBeforeKitKat(data); } } break;
在 handleImageOnKitKat(data) 中對(duì)地址進(jìn)行解析,根據(jù)三種不同的提供Uri方式采用不同的方法。
document 類型的 Uri
content 類型的 uri
file 類型的 Uri
@TargetApi(19) private void handleImageOnKitKat(Intent data) { // Toast.makeText(this,"到了handleImageOnKitKat(Intent data)方法了", Toast.LENGTH_LONG).show(); String imagePath = null; Uri uri = data.getData(); if(DocumentsContract.isDocumentUri(this, uri)){ //如果是 document 類型的 Uri,則通過 document id 處理 String docId = DocumentsContract.getDocumentId(uri); if("com.android.providers.media.documents".equals(uri.getAuthority())){ String id = docId.split(":")[1];//解析出數(shù)字格式的 id String selection = MediaStore.Images.Media._ID + "=" + id; imagePath = 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)); imagePath = getImagePath(contentUri, null); } }else if ("content".equalsIgnoreCase(uri.getScheme())){ //如果是 content 類型的 uri , 則使用普通方式處理 imagePath = getImagePath(uri, null); }else if("file".equalsIgnoreCase(uri.getScheme())){ //如果是 file 類型的 Uri,直接獲取圖片路徑即可 imagePath = uri.getPath(); } displayImage(imagePath);//顯示選中的圖片 }
注意那個(gè) @TargetApi(19) ,因?yàn)槲覀兊捻?xiàng)目最小兼容系統(tǒng)設(shè)置的是 15,而代碼中的 DocumentsContract.isDocumentUri 和 DocumentsContract.getDocumentId 有系統(tǒng)兼容性需要處理。
當(dāng)我們通過各個(gè)途徑,已經(jīng)獲取到圖片路徑的之后,自然就是顯示圖片了。于是最后一句代碼就是調(diào)用 displayImage 方法來實(shí)現(xiàn)圖片的顯示了。
對(duì)于 handeleImageBeforeKitKat 就要簡(jiǎn)單的多了。直接取得地址顯示。
private void handeleImageBeforeKitKat(Intent data){ Uri uri = data.getData(); String imagePath = getImagePath(uri, null); displayImage(imagePath); }
中間用到的獲得圖片真實(shí)路徑和顯示圖片的方法如下:
private String getImagePath(Uri uri, String selection) { String path = null; //通過 Uri 和 selection 來獲取真實(shí)的圖片路徑 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; } private void displayImage(String imagePath) { if(imagePath != null){ Bitmap bitmap = BitmapFactory.decodeFile(imagePath); picture.setImageBitmap(bitmap); }else{ Toast.makeText(this,"failed to get image", Toast.LENGTH_LONG).show(); } }
MainActivity.java完整代碼:
package com.cofox.mycameraalbum; import android.Manifest; import android.annotation.TargetApi; import android.content.ContentUris; import android.content.Intent; import android.content.pm.PackageManager; import android.database.Cursor; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.net.Uri; import android.os.Build; import android.provider.DocumentsContract; import android.provider.MediaStore; import android.support.annotation.NonNull; import android.support.v4.app.ActivityCompat; import android.support.v4.content.ContextCompat; import android.support.v4.content.FileProvider; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; import android.view.View; import android.widget.Button; import android.widget.ImageView; import android.widget.Toast; import java.io.File; import java.io.FileNotFoundException; import java.io.IOException; public class MainActivity extends AppCompatActivity { public static final int TAKE_PHOTO = 1; public static final int CHOOSE_PHOTO = 2; private ImageView picture; private Uri imageUri; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Button btnTakePhoto = (Button) findViewById(R.id.button_take_photo); //拍照按鈕 Button btnChooseFromAlbum = (Button)findViewById(R.id.button_choose_from_album);//相冊(cè)選擇圖片按鈕 picture = (ImageView) findViewById(R.id.photo_pictrue); //圖片控件,用來顯示照片的。 //打開相機(jī)拍照 btnTakePhoto.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { //創(chuàng)建File對(duì)象,用于存儲(chǔ)拍照后的圖片 File outputImage = new File(getExternalCacheDir(), "output_image.jpg"); try { if (outputImage.exists()) { outputImage.delete(); } outputImage.createNewFile(); } catch (IOException e) { e.printStackTrace(); } //android 7.0版本以下的系統(tǒng),直接Uri.fromFile取得真實(shí)文件路徑;7.0及以上版本的系統(tǒng),使用fileprovider封裝過的Uri再提供出去。 if (Build.VERSION.SDK_INT >= 24) { imageUri = FileProvider.getUriForFile(MainActivity.this, "com.cofox.mycameraalbum.fileprovider", outputImage); } else { imageUri = Uri.fromFile(outputImage); } //啟動(dòng)相機(jī)程序 Intent intent = new Intent("android.media.action.IMAGE_CAPTURE"); intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri); startActivityForResult(intent, TAKE_PHOTO); //啟動(dòng)Intent活動(dòng),拍完照會(huì)有結(jié)果返回到onActivityResult()方法中。 } }); //打開相冊(cè)選擇圖片 btnChooseFromAlbum.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { if(ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED){ ActivityCompat.requestPermissions(MainActivity.this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, 1); }else{ openAlbum(); } } }); } private void openAlbum() { Intent intent = new Intent("android.intent.action.GET_CONTENT"); intent.setType("image/*"); startActivityForResult(intent, CHOOSE_PHOTO);//打開相冊(cè) } @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { // super.onRequestPermissionsResult(requestCode, permissions, grantResults); switch (requestCode){ case 1: if(grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED){ openAlbum(); }else{ Toast.makeText(this, "You denied the permision.", Toast.LENGTH_LONG).show(); } break; default: } } @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { switch (requestCode) { case TAKE_PHOTO: if(resultCode == RESULT_OK){ try { //將拍攝的照片顯示出來 Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri)); picture.setImageBitmap(bitmap); } catch (FileNotFoundException e) { e.printStackTrace(); } } break; case CHOOSE_PHOTO: if(resultCode == RESULT_OK){ //判斷手機(jī)系統(tǒng)版本號(hào) if(Build.VERSION.SDK_INT >= 19){ //4.4及以上系統(tǒng)使用這個(gè)方法處理圖片 handleImageOnKitKat(data); }else{ //4.4以下系統(tǒng)使用這個(gè)方法處理圖片 handeleImageBeforeKitKat(data); } } break; default: break; } } @TargetApi(19) private void handleImageOnKitKat(Intent data) { // Toast.makeText(this,"到了handleImageOnKitKat(Intent data)方法了", Toast.LENGTH_LONG).show(); String imagePath = null; Uri uri = data.getData(); if(DocumentsContract.isDocumentUri(this, uri)){ //如果是 document 類型的 Uri,則通過 document id 處理 String docId = DocumentsContract.getDocumentId(uri); if("com.android.providers.media.documents".equals(uri.getAuthority())){ String id = docId.split(":")[1];//解析出數(shù)字格式的 id String selection = MediaStore.Images.Media._ID + "=" + id; imagePath = 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)); imagePath = getImagePath(contentUri, null); } }else if ("content".equalsIgnoreCase(uri.getScheme())){ //如果是 content 類型的 uri , 則使用普通方式處理 imagePath = getImagePath(uri, null); }else if("file".equalsIgnoreCase(uri.getScheme())){ //如果是 file 類型的 Uri,直接獲取圖片路徑即可 imagePath = uri.getPath(); } displayImage(imagePath);//顯示選中的圖片 } private void handeleImageBeforeKitKat(Intent data){ Uri uri = data.getData(); String imagePath = getImagePath(uri, null); displayImage(imagePath); } private String getImagePath(Uri uri, String selection) { String path = null; //通過 Uri 和 selection 來獲取真實(shí)的圖片路徑 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; } private void displayImage(String imagePath) { if(imagePath != null){ Bitmap bitmap = BitmapFactory.decodeFile(imagePath); picture.setImageBitmap(bitmap); }else{ Toast.makeText(this,"failed to get image", Toast.LENGTH_LONG).show(); } } }
看一下AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.cofox.mycameraalbum"> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> <application android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name" android:roundIcon="@mipmap/ic_launcher_round" android:supportsRtl="true" android:theme="@style/AppTheme"> <activity android:name=".MainActivity"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> <provider android:authorities="com.cofox.mycameraalbum.fileprovider" android:name="android.support.v4.content.FileProvider" android:exported="false" android:grantUriPermissions="true"> <meta-data android:name="android.support.FILE_PROVIDER_PATHS" android:resource="@xml/file_paths" /> </provider> </application> </manifest>
運(yùn)行效果圖片如下:
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
android實(shí)現(xiàn)手機(jī)傳感器調(diào)用
這篇文章主要為大家詳細(xì)介紹了android實(shí)現(xiàn)手機(jī)傳感器調(diào)用,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-04-04Android實(shí)現(xiàn)帶有進(jìn)度條的按鈕效果
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)帶有進(jìn)度條的按鈕效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-05-05Android利用Theme自定義Activity間的切換動(dòng)畫
這篇文章主要為大家詳細(xì)介紹了Android利用Theme自定義Activity間的切換動(dòng)畫,感興趣的小伙伴們可以參考一下2016-09-09Android之使用Android-query框架開發(fā)實(shí)戰(zhàn)(一)
這篇文章主要介紹了Android之使用Android-query框架開發(fā)實(shí)戰(zhàn)(一)的相關(guān)資料,需要的朋友可以參考下2015-10-10Android實(shí)現(xiàn)手機(jī)監(jiān)控?cái)z像頭
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)手機(jī)監(jiān)控?cái)z像頭,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03Android StickListView實(shí)現(xiàn)懸停效果
這篇文章主要介紹了Android StickListView實(shí)現(xiàn)懸停效果的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-06-06Android中的廣播、服務(wù)、數(shù)據(jù)庫、通知、包等術(shù)語的原理和介紹(圖解)
這篇文章主要介紹了Android中的廣播、服務(wù)、數(shù)據(jù)庫、通知、包等術(shù)語的原理和介紹(圖解),本文用圖文方式總結(jié)了Android中的一些開發(fā)術(shù)語,需要的朋友可以參考下2014-10-10