調用攝像頭和相冊完全解析

拍照獲取圖片流程

  1. 創建一個File對象,用於裝相機拍下的照片,起名:output_image.jpg
  2. 放到當前應用關聯緩存目錄下,getCacheDir()/getExternalCacheDir()
  3. 以android 7.0 爲界,用不同的方式將File轉換成Uri對象
  4. 然後構建Intent ,傳入Uri,隱式啓動相機
  5. onActivityResult()得到返回值並處理

創建File對象

先創建一個file

        //創建File對象,用於存儲拍照後的圖片
        File outputImage = new File(getDiskCacheDir(this), "output_img.jpg");
        //判斷該目錄下是否有文件存在
        try {
            if (outputImage.exists()) {
                outputImage.delete();
            }
            outputImage.createNewFile();
        } catch (IOException e) {
            e.printStackTrace();
        }

放到當前應用關聯緩存目錄

獲取可以使用的緩存地址:這裏以sd卡的緩存目錄爲優先,sd卡不能用就用手機本地緩存目錄。
- 應用關聯緩存目錄:SD卡中專門存放當前應用緩存數據的位置;
調用getExternalCacheDir()方法可以得到這個目錄,在卸載程序的時候,這裏數據可以被一起刪除,而且用緩存目錄的話即便是6.0以上的系統也不需要動態獲取讀寫權限,因爲不在內存而是緩存

    /**
     * 獲取緩存地址,如果sd卡不存在就獲取手機應用關聯緩存目錄
     * 卸載程序可被刪除,android6.0後不需要請求讀寫權限
     * getExternalCacheDir():/sdcard/Android/data/<application package>/cache
     * getCacheDir():/data/data/<application package>/cache
     * @param context
     * @return
     */
    public String getDiskCacheDir(Context context) {
        String cachePath = null;
        //Environment.getExtemalStorageState() 獲取SDcard的狀態
        //Environment.MEDIA_MOUNTED 只有改狀態下纔可以進行讀寫
        if (Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState())
                || !Environment.isExternalStorageRemovable()) {
            //sd卡應用關聯緩存目錄
            cachePath = context.getExternalCacheDir().getPath();
        } else {
            //手機應用關聯緩存目錄
            cachePath = context.getCacheDir().getPath();
        }
        return cachePath;
    }

將File轉換成Uri對象

啓動相機需要傳入有效的Uri才行,現在還是File格式,需要轉換。
- Android7.0 開始,直接使用本地真實路徑的Uri被認爲是不安全的,會拋出FileUriException異常。而FileProvider則是一種特殊的內容提供器,它使用了和內容提供器類似的機制來對數據進行保護。

        //Android 7.0 開始,直接使用本地真實路徑的Uri被認爲是不安全的,而FileProvide則是一種特殊的內容提供器
        if (Build.VERSION.SDK_INT >= 24) {
            imgUri = FileProvider.getUriForFile(MainActivity.this, "com.yuanli.choosephoto.fileprovider", outputImage);
        } else {
            imgUri = Uri.fromFile(outputImage);
        }

構建Intent ,傳入Uri,隱式啓動相機

        //隱式啓動相機程序
        Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");
        intent.putExtra(MediaStore.EXTRA_OUTPUT, imgUri);
        startActivityForResult(intent, TAKE_PHOTO);

onActivityResult()得到返回值並處理

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        if (resultCode == Activity.RESULT_CANCELED) {
            return;
        }
        switch (requestCode) {
            case TAKE_PHOTO:
                if (resultCode == RESULT_OK) {
                    //將拍攝的照片展示出來
                    try {
                        Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imgUri));
                        imgChoose.setImageBitmap(bitmap);
                    } catch (FileNotFoundException e) {
                        e.printStackTrace();
                    }
                }
                break;
            default:
                break;
        }
    }

寫着寫着跑偏了,研究相冊的時候需要動態申請權限,就想封裝一個好點的,結果看各種文檔一天就有過去了。還好差不多了。

被註解難住了,好難啊!看了各種文檔了,想自己寫不想用hy大神的庫,可仿不出hy大神的註解,好狗的難,到底哪裏沒想明白???
過了個週末。。。。。那麼久,,,,,繼續寫。。。

對不起,自己勉強明白了個大概,粗略的寫在了下一篇博客裏。不值得關注,有需要的還是自行查閱比較好。等我真正消化了,再好好真理一篇清晰易懂的。

繼續相機

相冊選取圖片流程

  1. 獲取讀寫權限
  2. 打開圖庫
  3. 處理返回結果

打開圖庫

獲取權限有很多方式,自己選擇喜歡的,我就直接說正題了

   private void choosePhoto() {
        //隱式啓動相冊
        Intent intent = new Intent("android.intent.action.GET_CONTENT");
        intent.setType("image/*");
        startActivityForResult(intent, CHOOSE_PHOTO);
    }

處理返回結果

啓動Intent時傳入數據是Uri格式,而Intent返回結果也是Uri格式的。只不過老規矩,4.4版本(19)爲界,4.4以上返回的Uri不是圖片真實的Uri,而是的封裝過的。這裏挺奇怪哈,拍照的時候給Intent傳入Uri的時候,界限是7.0(24);不懂咋想的。反正記住別混爲一談了。

返回值判斷:

            case CHOOSE_PHOTO:
                if(resultCode==RESULT_OK){
                    //判斷手機系統版本號
                    if(Build.VERSION.SDK_INT>=Build.VERSION_CODES.KITKAT){
                        //4.4ji以上系統使用這個方法處理圖片
                        handlerImageOnKitKat(data);
                    }else{
                        //4.4ji以下系統使用這個方法處理圖片
                        handlerImageBeforeKitKat(data);
                    }
                }
                break;
    /**
     * android系統從4.4版本開始,選取相冊中圖片不再返回圖片的真實的Uri了,
     * 而是一個封裝過的Uri,因此需要對這個Uri進行解析才行。
     * @param data
     */
    @RequiresApi(api = Build.VERSION_CODES.KITKAT)
    private void handlerImageOnKitKat(Intent data) {
        String imagePath=null;
        Uri uri=data.getData();
        imagePath = uriToPath(uri);
//        startPhotoZoom(new File(imagePath), 350);
        displayImage(imagePath);
    }

    private String uriToPath(Uri uri) {
        String path = null;
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
            if (DocumentsContract.isDocumentUri(this, uri)) {
                // 如果是document類型的Uri,則通過document id處理
                String docId = DocumentsContract.getDocumentId(uri);
                //如果authority是media格式的話,document id 還需要在進行一次解析
                if ("com.android.providers.media.documents".equals(uri.getAuthority())) {
                    String id = docId.split(":")[1]; // 解析出真正的數字格式的id
                    String selection = MediaStore.Images.Media._ID + "=" + id;
                    path = getImagePath(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, selection);
                } else if ("com.android.providers.downloads.documents".equals(uri.getAuthority())) {
                    Uri contentUri = ContentUris.withAppendedId(Uri.parse("content://downloads/public_downloads"), Long.valueOf(docId));
                    path = getImagePath(contentUri, null);
                }
            } else if ("content".equalsIgnoreCase(uri.getScheme())) {
                // 如果是content類型的Uri,則使用普通方式處理
                path = getImagePath(uri, null);
            } else if ("file".equalsIgnoreCase(uri.getScheme())) {
                // 如果是file類型的Uri,直接獲取圖片路徑即可
                path = uri.getPath();
            }
        }
        return path;
    }

    private String getImagePath(Uri uri, String selection) {
        String path = null;
        // 通過Uri和selection來獲取真實的圖片路徑
        Cursor cursor = getContentResolver().query(uri, null, selection, null, null);
        if (cursor != null) {
            if (cursor.moveToFirst()) {
                path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA));
            }
            cursor.close();
        }
        return path;
    }
    /**
     * android系統從4.4版本一下執行
     * @param data
     */
    private void handlerImageBeforeKitKat(Intent data) {
        Uri uri = data.getData();
        String imagePath = getImagePath(uri, null);
        displayImage(imagePath);
        Log.i("TAG", "file://" + imagePath + "選擇圖片的URI" + uri);
//        startPhotoZoom(new File(imagePath), 350);
    }
    private void displayImage(String imagePath) {
        if(imagePath!=null){
            Bitmap bitmap=BitmapFactory.decodeFile(imagePath);
            imgChoose.setImageBitmap(bitmap);
        }
    }
發佈了48 篇原創文章 · 獲贊 12 · 訪問量 5萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章