Android拍照或從圖庫(kù)選擇圖片并裁剪
今天看《第一行代碼》上面關(guān)于拍照和從相冊(cè)選取圖片那一部分,發(fā)現(xiàn)始終出不來(lái)效果,所以搜索其他資料學(xué)習(xí)一下相關(guān)知識(shí),寫(xiě)一個(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寫(xiě)入SD卡中的一個(gè)文件中,并返回寫(xiě)入文件的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 {
//打開(kāi)文件輸出流
FileOutputStream fos = new FileOutputStream(img);
//將bitmap壓縮后寫(xiě)入輸出流(參數(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)裁剪程序
/**
* 通過(guò)Uri傳遞圖像信息以供裁剪
* @param uri
*/
private void startImageZoom(Uri uri){
//構(gòu)建隱式Intent來(lái)啟動(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ù)是通過(guò)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/*");
//打開(kāi)圖片選擇
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寫(xiě)入SD卡中的一個(gè)文件中,并返回寫(xiě)入文件的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 {
//打開(kāi)文件輸出流
FileOutputStream fos = new FileOutputStream(img);
//將bitmap壓縮后寫(xiě)入輸出流(參數(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;
}
}
/**
* 通過(guò)Uri傳遞圖像信息以供裁剪
* @param uri
*/
private void startImageZoom(Uri uri){
//構(gòu)建隱式Intent來(lái)啟動(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ù)是通過(guò)Intent返回
intent.putExtra("return-data", true);
startActivityForResult(intent, CROP_CODE);
}
}
注:最后還需要在AndroidManifest文件中加入存儲(chǔ)卡讀寫(xiě)權(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ò)圖片卻無(wú)法裁剪問(wèn)題的方法
- Android實(shí)現(xiàn)從本地圖庫(kù)/相機(jī)拍照后裁剪圖片并設(shè)置頭像
- Android 7.0中拍照和圖片裁剪適配的問(wèn)題詳解
- 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)求的三種方式,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2018-12-12
Android ViewGroup事件分發(fā)和處理源碼分析
這篇文章主要為大家介紹了Android ViewGroup事件分發(fā)和處理源碼分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-02-02
Android實(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-03
Android自定義View系列之Path繪制仿支付寶支付成功動(dòng)畫(huà)
這篇文章主要為大家詳細(xì)介紹了Android自定義View系列之Path繪制仿支付寶支付成功動(dòng)畫(huà),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-12-12
Android 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-05
Android開(kāi)發(fā)之完全隱藏軟鍵盤(pán)的方法
這篇文章主要介紹了Android開(kāi)發(fā)之完全隱藏軟鍵盤(pán)的方法的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-06-06
Android RecyclerView 上拉加載更多及下拉刷新功能的實(shí)現(xiàn)方法
這篇文章主要介紹了Android RecyclerView 上拉加載更多及下拉刷新的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-09-09

