1. 程式人生 > >Android拍照和獲取相簿圖片

Android拍照和獲取相簿圖片

之前遇到各種拍照啊,獲取相簿圖片之類,都是直接去度娘,要麼之前的程式碼複製下,沒好好總結過。

再也不要問度娘了,再也不用一堆部落格裡找啊找了。。。

----------------------------------------------我是正文的分割線-----------------------------------------------------------

一個一個來,先說呼叫手機相機拍照(最簡單版):

cameraButton.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View v) {
        Intent intent=new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        startActivityForResult(intent,TAKE_PHOTO);
    }
});

protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);
    if(resultCode==RESULT_OK){
        Bundle bundle=data.getExtras();
        Bitmap bitmap=(Bitmap)bundle.get("data");
        photoImageView.setImageBitmap(bitmap);
    }
}
這個方法簡單是簡單了,但是得到的圖片卻是縮圖,通常都非常模糊,很多時候並不滿足我們的要求,我們需要的是獲取拍照所得照片的原圖。

而通常想要獲取拍照所得原圖的方法,我們首先會自定義圖片名稱,確認圖片儲存位置,之後根據圖片位置Uri,自然可以獲得原圖Bitmap。程式碼如

    /**
     * 自定義圖片名,獲取照片的file
     */
    private File createImgFile(){
        //確定檔名
        String fileName="img_"+new SimpleDateFormat("yyyyMMdd_HHmmss").format(new Date())+".jpg";
//        File dir=getExternalFilesDir(Environment.DIRECTORY_PICTURES);
//        File dir=Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES);
//        File dir=Environment.getExternalStorageDirectory();
        File dir;
        if(Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)){
            dir=Environment.getExternalStorageDirectory();
        }else{
            dir=getExternalFilesDir(Environment.DIRECTORY_PICTURES);
        }
        File tempFile=new File(dir,fileName);
        try{
            if(tempFile.exists()){
                tempFile.delete();
            }
            tempFile.createNewFile();
        }catch (IOException e){
            e.printStackTrace();
        }
        //獲取檔案路徑
        photoPath=tempFile.getAbsolutePath();
        return tempFile;
    }
    //拍照
    cameraButton.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View view) {
            //只是加了一個uri作為地址傳入
            Intent intent=new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
            File photoFile=createImgFile();
            photoUri=Uri.fromFile(photoFile);
            intent.putExtra(MediaStore.EXTRA_OUTPUT,photoUri);
            startActivityForResult(intent,TAKE_PHOTO);
        }
    });

程式碼應該非常清楚,我們定義了圖片名稱,將uri傳入,那麼我們在onActivityResult中就可以得到這個uri,之後豈不是可以直接根據uri得到bitmap了?例如這樣:

Bitmap bitmap=MediaStore.Images.Media.getBitmap(getContentResolver(),photoUri);
photoImageView.setImageBitmap(bitmap);
這樣做起來是沒錯的,可惜通常我們手機的攝像頭大多非常高清,拍攝出來的照片如果直接載入到手機記憶體裡面,恐怕就啦啦啦啦啦啦了……

那麼,我們唯一能想到的,就只有開始壓縮圖片了:
    /**
     * 壓縮圖片
     */
    private void setImageBitmap(){
        //獲取imageview的寬和高
        int targetWidth=photoImageView.getWidth();
        int targetHeight=photoImageView.getHeight();

        //根據圖片路徑,獲取bitmap的寬和高
        BitmapFactory.Options options=new BitmapFactory.Options();
        options.inJustDecodeBounds=true;
        BitmapFactory.decodeFile(photoPath,options);
        int photoWidth=options.outWidth;
        int photoHeight=options.outHeight;

        //獲取縮放比例
        int inSampleSize=1;
        if(photoWidth>targetWidth||photoHeight>targetHeight){
            int widthRatio=Math.round((float)photoWidth/targetWidth);
            int heightRatio=Math.round((float)photoHeight/targetHeight);
            inSampleSize=Math.min(widthRatio,heightRatio);
        }

        //使用現在的options獲取Bitmap
        options.inSampleSize=inSampleSize;
        options.inJustDecodeBounds=false;
        Bitmap bitmap=BitmapFactory.decodeFile(photoPath,options);
        photoImageView.setImageBitmap(bitmap);
    }
以上就是調取相機拍照的全部程式碼,當然WRITE_EXTERNAL_STORAGE這個許可權肯定是要加的。

還有一件經常出現的事,就是可能你拍照之後,照片並沒有及時出現在手機相簿中,而是需要手機重啟之後才會出現。其實我們只需要傳送一條廣播,就可以將圖片新增進手機相簿,程式碼如下:

    //將圖片新增進手機相簿
    private void galleryAddPic(){
        Intent mediaScanIntent=new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);
        mediaScanIntent.setData(photoUri);
        this.sendBroadcast(mediaScanIntent);
    }
困難已經過去,相對而言,從相簿裡選取照片,簡直再簡單不過,我相信只要看下程式碼就會懂了,不信你看:
    //從相簿獲取照片
    albumButton.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View view) {
            //Intent intent=new Intent(Intent.ACTION_PICK);
            Intent intent=new Intent(Intent.ACTION_GET_CONTENT);
            intent.setType("image/*");
            startActivityForResult(intent,PICK_PHOTO);
        }
    });
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if(resultCode==RESULT_OK){
            switch (requestCode){
                case TAKE_PHOTO:
                    setImageBitmap();
                    galleryAddPic();
                    break;
                case PICK_PHOTO:
                    //data中自帶有返回的uri
                    photoUri=data.getData();
                    //獲取照片路徑
                    String[] filePathColumn={MediaStore.Audio.Media.DATA};
                    Cursor cursor=getContentResolver().query(photoUri,filePathColumn,null,null,null);
                    cursor.moveToFirst();
                    photoPath=cursor.getString(cursor.getColumnIndex(filePathColumn[0]));
                    cursor.close();
                    //有了照片路徑,之後就是壓縮圖片,和之前沒有什麼區別
                    setImageBitmap();
                    break;
            }
        }
    }
很簡單吧,我們在onActivityResult()中根據data獲取到了所選圖片的uri,之後只要一步查詢,就可以得到圖片路徑。而有了照片路徑,就沒有任何問題了。

整個demo到這裡就結束了,接下來的內容可以不必看,demo下載連結:下載地址

自問自答時間:

問:SD卡是什麼?因為在我建立File的時候,目錄檔案預設使用的是Environment.getExternalStorageDirectory(),而在使用之前是需要檢測狀態的,

也就是這句Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED),詢問SD卡是否正確安裝。我用了一部新手機,依舊是正確安裝的。

答:手機是有內建sd卡的,所以哪怕我買了新手機,但是依舊可以檢測到sd卡的存在。

一個有意思的回答是說,內建的sd卡類似於電腦硬碟,外界的sd卡類似於行動硬碟,但也不確定是否正確。

問:建立File的時候,幾個不同的目錄檔案分別什麼意思?有什麼不同?

答:直接看測試結果吧,Environment.getExternalStorageDirectory()的地址:/storage/emulated/0

Environment.getExternalStoragePublicDirectory()的地址:/storage/emulated/0/Pictures

getExternalFilesDir()的地址:/storage/emulated/0/Android/data/com.example.notificationapp/files/Pictures

問:從相簿選取照片的時候,好像有個兩個Intent都可以,有什麼區別?

答:Intent intent=new Intent(Intent.ACTION_PICK);
Intent intent=new Intent(Intent.ACTION_GET_CONTENT);

這兩個確實都可以從相簿選取照片,但是具體有什麼區別,好吧,我沒搞明白。