Android拍照或從圖庫選擇圖片并裁剪
今天看《第一行代碼》上面關(guān)于拍照和從相冊選取圖片那一部分,發(fā)現(xiàn)始終出不來效果,所以搜索其他資料學習一下相關(guān)知識,寫一個簡單的Demo。
一、 拍照選擇圖片
1、使用隱式Intent啟動相機
//構(gòu)建隱式Intent Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); //調(diào)用系統(tǒng)相機 startActivityForResult(intent, 1);
2、處理相機拍照返回的結(jié)果
//用戶點擊了取消 if(data == null){ return; }else{ Bundle extras = data.getExtras(); if (extras != null){ //獲得拍的照片 Bitmap bm = extras.getParcelable("data"); } }
二、 從圖庫選擇圖片
1、構(gòu)建內(nèi)容選擇隱式Intent
Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
2、設置內(nèi)容類型為圖片
intent.setType("image/*");
3、啟動圖片選擇
startActivityForResult(intent, 2);
4、處理圖片選擇結(jié)果
if (data == null){ return; }else{ //用戶從圖庫選擇圖片后會返回所選圖片的Uri Uri uri; //獲取到用戶所選圖片的Uri uri = data.getData(); }
三、 裁剪選擇的圖片
從相機拍照得到的是Bitmap類型,所以我們需要先將其轉(zhuǎn)化為文件Uri以供裁剪。同時我們還要順便將相機拍的照片保存到本地。
/** * 將Bitmap寫入SD卡中的一個文件中,并返回寫入文件的Uri * @param bm * @param dirPath * @return */ private Uri saveBitmap(Bitmap bm, String dirPath) { //新建文件夾用于存放裁剪后的圖片 File tmpDir = new File(Environment.getExternalStorageDirectory() + "/" + dirPath); if (!tmpDir.exists()){ tmpDir.mkdir(); } //新建文件存儲裁剪后的圖片 File img = new File(tmpDir.getAbsolutePath() + "/avator.png"); try { //打開文件輸出流 FileOutputStream fos = new FileOutputStream(img); //將bitmap壓縮后寫入輸出流(參數(shù)依次為圖片格式、圖片質(zhì)量和輸出流) bm.compress(Bitmap.CompressFormat.PNG, 85, fos); //刷新輸出流 fos.flush(); //關(guān)閉輸出流 fos.close(); //返回File類型的Uri return Uri.fromFile(img); } catch (FileNotFoundException e) { e.printStackTrace(); return null; } catch (IOException e) { e.printStackTrace(); return null; } }
從圖庫選擇的圖片返回的是content類型的Uri,我們需要轉(zhuǎn)化為文件類型的Uri才能進行裁剪。
/** * 將content類型的Uri轉(zhuǎn)化為文件類型的Uri * @param uri * @return */ private Uri convertUri(Uri uri){ InputStream is; try { //Uri ----> InputStream is = getContentResolver().openInputStream(uri); //InputStream ----> Bitmap Bitmap bm = BitmapFactory.decodeStream(is); //關(guān)閉流 is.close(); return saveBitmap(bm, "temp"); } catch (FileNotFoundException e) { e.printStackTrace(); return null; } catch (IOException e) { e.printStackTrace(); return null; } }
下面便是設置裁剪的參數(shù),用隱式Intent方式啟動裁剪程序
/** * 通過Uri傳遞圖像信息以供裁剪 * @param uri */ private void startImageZoom(Uri uri){ //構(gòu)建隱式Intent來啟動裁剪程序 Intent intent = new Intent("com.android.camera.action.CROP"); //設置數(shù)據(jù)uri和類型為圖片類型 intent.setDataAndType(uri, "image/*"); //顯示View為可裁剪的 intent.putExtra("crop", true); //裁剪的寬高的比例為1:1 intent.putExtra("aspectX", 1); intent.putExtra("aspectY", 1); //輸出圖片的寬高均為150 intent.putExtra("outputX", 150); intent.putExtra("outputY", 150); //裁剪之后的數(shù)據(jù)是通過Intent返回 intent.putExtra("return-data", true); startActivityForResult(intent, CROP_CODE); }
下面是完整的布局文件和java文件
--------------------------------------------------------------------------------
activity_main.xml文件
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical"> <ImageView android:id="@+id/show_image" android:layout_width="match_parent" android:layout_height="300dp"/> <Button android:id="@+id/choose_camera" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="攝像頭"/> <Button android:id="@+id/choose_gallery" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="相冊"/> </LinearLayout>
MainActivity.java文件
public class MainActivity extends AppCompatActivity implements View.OnClickListener { //用于展示選擇的圖片 private ImageView mImageView; private static final int CAMERA_CODE = 1; private static final int GALLERY_CODE = 2; private static final int CROP_CODE = 3; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); initView(); } private void initView() { mImageView = (ImageView) findViewById(R.id.show_image); Button chooseCamera = (Button) findViewById(R.id.choose_camera); chooseCamera.setOnClickListener(this); Button chooseGallery = (Button) findViewById(R.id.choose_gallery); chooseGallery.setOnClickListener(this); } @Override public void onClick(View view) { switch (view.getId()){ case R.id.choose_camera: //拍照選擇 chooseFromCamera(); break; case R.id.choose_gallery: //從相冊選取 chooseFromGallery(); break; default: break; } } /** * 拍照選擇圖片 */ private void chooseFromCamera() { //構(gòu)建隱式Intent Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); //調(diào)用系統(tǒng)相機 startActivityForResult(intent, CAMERA_CODE); } /** * 從相冊選擇圖片 */ private void chooseFromGallery() { //構(gòu)建一個內(nèi)容選擇的Intent Intent intent = new Intent(Intent.ACTION_GET_CONTENT); //設置選擇類型為圖片類型 intent.setType("image/*"); //打開圖片選擇 startActivityForResult(intent, GALLERY_CODE); } @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { switch (requestCode){ case CAMERA_CODE: //用戶點擊了取消 if(data == null){ return; }else{ Bundle extras = data.getExtras(); if (extras != null){ //獲得拍的照片 Bitmap bm = extras.getParcelable("data"); //將Bitmap轉(zhuǎn)化為uri Uri uri = saveBitmap(bm, "temp"); //啟動圖像裁剪 startImageZoom(uri); } } break; case GALLERY_CODE: if (data == null){ return; }else{ //用戶從圖庫選擇圖片后會返回所選圖片的Uri Uri uri; //獲取到用戶所選圖片的Uri uri = data.getData(); //返回的Uri為content類型的Uri,不能進行復制等操作,需要轉(zhuǎn)換為文件Uri uri = convertUri(uri); startImageZoom(uri); } break; case CROP_CODE: if (data == null){ return; }else{ Bundle extras = data.getExtras(); if (extras != null){ //獲取到裁剪后的圖像 Bitmap bm = extras.getParcelable("data"); mImageView.setImageBitmap(bm); } } break; default: break; } } /** * 將content類型的Uri轉(zhuǎn)化為文件類型的Uri * @param uri * @return */ private Uri convertUri(Uri uri){ InputStream is; try { //Uri ----> InputStream is = getContentResolver().openInputStream(uri); //InputStream ----> Bitmap Bitmap bm = BitmapFactory.decodeStream(is); //關(guān)閉流 is.close(); return saveBitmap(bm, "temp"); } catch (FileNotFoundException e) { e.printStackTrace(); return null; } catch (IOException e) { e.printStackTrace(); return null; } } /** * 將Bitmap寫入SD卡中的一個文件中,并返回寫入文件的Uri * @param bm * @param dirPath * @return */ private Uri saveBitmap(Bitmap bm, String dirPath) { //新建文件夾用于存放裁剪后的圖片 File tmpDir = new File(Environment.getExternalStorageDirectory() + "/" + dirPath); if (!tmpDir.exists()){ tmpDir.mkdir(); } //新建文件存儲裁剪后的圖片 File img = new File(tmpDir.getAbsolutePath() + "/avator.png"); try { //打開文件輸出流 FileOutputStream fos = new FileOutputStream(img); //將bitmap壓縮后寫入輸出流(參數(shù)依次為圖片格式、圖片質(zhì)量和輸出流) bm.compress(Bitmap.CompressFormat.PNG, 85, fos); //刷新輸出流 fos.flush(); //關(guān)閉輸出流 fos.close(); //返回File類型的Uri return Uri.fromFile(img); } catch (FileNotFoundException e) { e.printStackTrace(); return null; } catch (IOException e) { e.printStackTrace(); return null; } } /** * 通過Uri傳遞圖像信息以供裁剪 * @param uri */ private void startImageZoom(Uri uri){ //構(gòu)建隱式Intent來啟動裁剪程序 Intent intent = new Intent("com.android.camera.action.CROP"); //設置數(shù)據(jù)uri和類型為圖片類型 intent.setDataAndType(uri, "image/*"); //顯示View為可裁剪的 intent.putExtra("crop", true); //裁剪的寬高的比例為1:1 intent.putExtra("aspectX", 1); intent.putExtra("aspectY", 1); //輸出圖片的寬高均為150 intent.putExtra("outputX", 150); intent.putExtra("outputY", 150); //裁剪之后的數(shù)據(jù)是通過Intent返回 intent.putExtra("return-data", true); startActivityForResult(intent, CROP_CODE); } }
注:最后還需要在AndroidManifest文件中加入存儲卡讀寫權(quán)限
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
- Android實現(xiàn)拍照、選擇圖片并裁剪圖片功能
- Android裁剪圖片為圓形圖片的實現(xiàn)原理與代碼
- 解決Android從相冊中獲取圖片出錯圖片卻無法裁剪問題的方法
- Android實現(xiàn)從本地圖庫/相機拍照后裁剪圖片并設置頭像
- Android 7.0中拍照和圖片裁剪適配的問題詳解
- Android自定義View實現(xiàn)照片裁剪框與照片裁剪功能
- Android實現(xiàn)拍照及圖片裁剪(6.0以上權(quán)限處理及7.0以上文件管理)
- Android編程實現(xiàn)調(diào)用系統(tǒng)圖庫與裁剪圖片功能
- Android圖片裁剪功能實現(xiàn)代碼
- android實現(xiàn)拖拽裁剪功能
相關(guān)文章
Android ViewGroup事件分發(fā)和處理源碼分析
這篇文章主要為大家介紹了Android ViewGroup事件分發(fā)和處理源碼分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-02-02Android實現(xiàn)自動變換大小的組件ViewPager2
這篇文章主要介紹了Android實現(xiàn)自動變換大小的組件ViewPager2,ViewPager2最顯著的特點是基于RecyclerView實現(xiàn),RecyclerView是目前Android端最成熟的AdapterView解決方案2023-03-03Android自定義View系列之Path繪制仿支付寶支付成功動畫
這篇文章主要為大家詳細介紹了Android自定義View系列之Path繪制仿支付寶支付成功動畫,具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-12-12基于Android中dp和px之間進行轉(zhuǎn)換的實現(xiàn)代碼
本篇文章是對在Android中dp和px之間進行轉(zhuǎn)換的實現(xiàn)方法進行了分析介紹,需要的朋友參考下2013-05-05Android RecyclerView 上拉加載更多及下拉刷新功能的實現(xiàn)方法
這篇文章主要介紹了Android RecyclerView 上拉加載更多及下拉刷新的相關(guān)資料,非常不錯,具有參考借鑒價值,需要的朋友可以參考下2016-09-09