Android拍照或從相冊(cè)選擇圖片并裁剪

今天看《第一行代碼》上面關(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");
    }
}

二钩蚊、 從圖庫選擇圖片

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{
    //用戶從圖庫選擇圖片后會(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;
    }
}

從圖庫選擇的圖片返回的是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{
                    //用戶從圖庫選擇圖片后會(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.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市鉴逞,隨后出現(xiàn)的幾起案子记某,更是在濱河造成了極大的恐慌,老刑警劉巖华蜒,帶你破解...
    沈念sama閱讀 219,188評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件辙纬,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡叭喜,警方通過查閱死者的電腦和手機(jī)贺拣,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,464評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來捂蕴,“玉大人譬涡,你說我怎么就攤上這事∩侗妫” “怎么了涡匀?”我有些...
    開封第一講書人閱讀 165,562評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)溉知。 經(jīng)常有香客問我陨瘩,道長(zhǎng),這世上最難降的妖魔是什么级乍? 我笑而不...
    開封第一講書人閱讀 58,893評(píng)論 1 295
  • 正文 為了忘掉前任舌劳,我火速辦了婚禮,結(jié)果婚禮上玫荣,老公的妹妹穿的比我還像新娘甚淡。我一直安慰自己,他們只是感情好捅厂,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,917評(píng)論 6 392
  • 文/花漫 我一把揭開白布贯卦。 她就那樣靜靜地躺著资柔,像睡著了一般。 火紅的嫁衣襯著肌膚如雪撵割。 梳的紋絲不亂的頭發(fā)上贿堰,一...
    開封第一講書人閱讀 51,708評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音睁枕,去河邊找鬼官边。 笑死,一個(gè)胖子當(dāng)著我的面吹牛外遇,可吹牛的內(nèi)容都是我干的注簿。 我是一名探鬼主播,決...
    沈念sama閱讀 40,430評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼跳仿,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼诡渴!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起菲语,我...
    開封第一講書人閱讀 39,342評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤妄辩,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后山上,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體眼耀,經(jīng)...
    沈念sama閱讀 45,801評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,976評(píng)論 3 337
  • 正文 我和宋清朗相戀三年佩憾,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了哮伟。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,115評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡妄帘,死狀恐怖楞黄,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情抡驼,我是刑警寧澤鬼廓,帶...
    沈念sama閱讀 35,804評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站致盟,受9級(jí)特大地震影響碎税,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜馏锡,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,458評(píng)論 3 331
  • 文/蒙蒙 一雷蹂、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧眷篇,春花似錦萎河、人聲如沸荔泳。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,008評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至昧港,卻和暖如春擎椰,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背创肥。 一陣腳步聲響...
    開封第一講書人閱讀 33,135評(píng)論 1 272
  • 我被黑心中介騙來泰國打工达舒, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人叹侄。 一個(gè)月前我還...
    沈念sama閱讀 48,365評(píng)論 3 373
  • 正文 我出身青樓巩搏,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國和親趾代。 傳聞我的和親對(duì)象是個(gè)殘疾皇子贯底,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,055評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容