Android拍照或從圖庫(kù)選擇圖片并裁剪
今天看《第一行代碼》上面關(guān)于拍照和從相冊(cè)選取圖片那一部分,發(fā)現(xiàn)始終出不來效果,所以搜索其他資料學(xué)習(xí)一下相關(guān)知識(shí),寫一個(gè)簡(jiǎn)單的Demo。
一、 拍照選擇圖片
1、使用隱式Intent啟動(dòng)相機(jī)
//構(gòu)建隱式Intent Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); //調(diào)用系統(tǒng)相機(jī) startActivityForResult(intent, 1);
2、處理相機(jī)拍照返回的結(jié)果
//用戶點(diǎn)擊了取消 if(data == null){ return; }else{ Bundle extras = data.getExtras(); if (extras != null){ //獲得拍的照片 Bitmap bm = extras.getParcelable("data"); } }
二、 從圖庫(kù)選擇圖片
1、構(gòu)建內(nèi)容選擇隱式Intent
Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
2、設(shè)置內(nèi)容類型為圖片
intent.setType("image/*");
3、啟動(dòng)圖片選擇
startActivityForResult(intent, 2);
4、處理圖片選擇結(jié)果
if (data == null){ return; }else{ //用戶從圖庫(kù)選擇圖片后會(huì)返回所選圖片的Uri Uri uri; //獲取到用戶所選圖片的Uri uri = data.getData(); }
三、 裁剪選擇的圖片
從相機(jī)拍照得到的是Bitmap類型,所以我們需要先將其轉(zhuǎn)化為文件Uri以供裁剪。同時(shí)我們還要順便將相機(jī)拍的照片保存到本地。
/** * 將Bitmap寫入SD卡中的一個(gè)文件中,并返回寫入文件的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(); } //新建文件存儲(chǔ)裁剪后的圖片 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; } }
從圖庫(kù)選擇的圖片返回的是content類型的Uri,我們需要轉(zhuǎn)化為文件類型的Uri才能進(jìn)行裁剪。
/** * 將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è)置裁剪的參數(shù),用隱式Intent方式啟動(dòng)裁剪程序
/** * 通過Uri傳遞圖像信息以供裁剪 * @param uri */ private void startImageZoom(Uri uri){ //構(gòu)建隱式Intent來啟動(dòng)裁剪程序 Intent intent = new Intent("com.android.camera.action.CROP"); //設(shè)置數(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="相冊(cè)"/> </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: //從相冊(cè)選取 chooseFromGallery(); break; default: break; } } /** * 拍照選擇圖片 */ private void chooseFromCamera() { //構(gòu)建隱式Intent Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); //調(diào)用系統(tǒng)相機(jī) startActivityForResult(intent, CAMERA_CODE); } /** * 從相冊(cè)選擇圖片 */ private void chooseFromGallery() { //構(gòu)建一個(gè)內(nèi)容選擇的Intent Intent intent = new Intent(Intent.ACTION_GET_CONTENT); //設(shè)置選擇類型為圖片類型 intent.setType("image/*"); //打開圖片選擇 startActivityForResult(intent, GALLERY_CODE); } @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { switch (requestCode){ case CAMERA_CODE: //用戶點(diǎn)擊了取消 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"); //啟動(dòng)圖像裁剪 startImageZoom(uri); } } break; case GALLERY_CODE: if (data == null){ return; }else{ //用戶從圖庫(kù)選擇圖片后會(huì)返回所選圖片的Uri Uri uri; //獲取到用戶所選圖片的Uri uri = data.getData(); //返回的Uri為content類型的Uri,不能進(jìn)行復(fù)制等操作,需要轉(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卡中的一個(gè)文件中,并返回寫入文件的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(); } //新建文件存儲(chǔ)裁剪后的圖片 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來啟動(dòng)裁剪程序 Intent intent = new Intent("com.android.camera.action.CROP"); //設(shè)置數(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文件中加入存儲(chǔ)卡讀寫權(quán)限
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android實(shí)現(xiàn)拍照、選擇圖片并裁剪圖片功能
- Android裁剪圖片為圓形圖片的實(shí)現(xiàn)原理與代碼
- 解決Android從相冊(cè)中獲取圖片出錯(cuò)圖片卻無法裁剪問題的方法
- Android實(shí)現(xiàn)從本地圖庫(kù)/相機(jī)拍照后裁剪圖片并設(shè)置頭像
- Android 7.0中拍照和圖片裁剪適配的問題詳解
- Android自定義View實(shí)現(xiàn)照片裁剪框與照片裁剪功能
- Android實(shí)現(xiàn)拍照及圖片裁剪(6.0以上權(quán)限處理及7.0以上文件管理)
- Android編程實(shí)現(xiàn)調(diào)用系統(tǒng)圖庫(kù)與裁剪圖片功能
- Android圖片裁剪功能實(shí)現(xiàn)代碼
- android實(shí)現(xiàn)拖拽裁剪功能
相關(guān)文章
安卓GET與POST網(wǎng)絡(luò)請(qǐng)求的三種方式
今天小編就為大家分享一篇關(guān)于安卓GET與POST網(wǎng)絡(luò)請(qǐng)求的三種方式,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧2018-12-12Android ViewGroup事件分發(fā)和處理源碼分析
這篇文章主要為大家介紹了Android ViewGroup事件分發(fā)和處理源碼分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-02-02Android實(shí)現(xiàn)自動(dòng)變換大小的組件ViewPager2
這篇文章主要介紹了Android實(shí)現(xiàn)自動(dòng)變換大小的組件ViewPager2,ViewPager2最顯著的特點(diǎn)是基于RecyclerView實(shí)現(xiàn),RecyclerView是目前Android端最成熟的AdapterView解決方案2023-03-03Android自定義View系列之Path繪制仿支付寶支付成功動(dòng)畫
這篇文章主要為大家詳細(xì)介紹了Android自定義View系列之Path繪制仿支付寶支付成功動(dòng)畫,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-12-12Android Studio實(shí)現(xiàn)進(jìn)度條效果
這篇文章主要為大家詳細(xì)介紹了Android Studio實(shí)現(xiàn)進(jìn)度條效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-04-04基于Android中dp和px之間進(jìn)行轉(zhuǎn)換的實(shí)現(xiàn)代碼
本篇文章是對(duì)在Android中dp和px之間進(jìn)行轉(zhuǎn)換的實(shí)現(xiàn)方法進(jìn)行了分析介紹,需要的朋友參考下2013-05-05Android RecyclerView 上拉加載更多及下拉刷新功能的實(shí)現(xiàn)方法
這篇文章主要介紹了Android RecyclerView 上拉加載更多及下拉刷新的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-09-09