Android開發在Activity外申請許可權呼叫相機開啟相簿

MicroDeLe發表於2021-08-01

問題描述:

  最近在專案中遇到一個需要呼叫相簿和開啟相機的需求,但是,在Android 6.0以後,呼叫相簿屬於危險許可權,需要開發者動態獲取,這就意味著我們申請許可權是與Activity繫結的,但如果一個App中需要多個地方請求開啟相簿,那我們要每個地方都要寫一遍開啟相簿的程式嗎(當然你可以Ctrl c v),但是,這對於任何一個有追求的程式設計師來說,都是不恰當的,所以我們要定義一個公共介面,做到在任何一個需要呼叫開啟相簿的地方隨時呼叫,增加了程式碼的複用性。好記性不如爛筆頭,把這個過程記錄下來,let's go!!

解決思路:

  既然申請許可權是與Activity繫結的,那麼我們就建立一個的Activity專門用於完成開啟相簿/相機,申請許可權的任務,當其它Activity需要用到這個功能時,直接跳轉到這個Activity,完成任務後,再返回照片的真實路徑就行了,Ok,思路有了,話不多說,直接擼碼。

1. 首先建立一個Activity專門用於申請許可權和開啟相簿相機的功能,這裡我命名為 SelectImageActivity

開啟相機的程式如下:第一個方法是開啟相機,第二個方法是建立一個圖片檔案,這種寫法是Android官方寫法,這裡定義了一個 currentPhotoPath 用來儲存圖片的路徑,後面需要返回給呼叫它的ativity

// 呼叫相機
    private void selectImageFromCamera() {
        Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);  // 初始化開啟相機的意圖
        if (takePictureIntent.resolveActivity(getPackageManager()) != null) {
            File photoFile = null;
            try {
                photoFile = createImageFile();
            } catch (IOException e) {
                e.printStackTrace();
            }
            if (photoFile != null) {
                Uri photoURI = FileProvider.getUriForFile(this, "com.example.encryption.fileProvider", photoFile);
                takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoURI);
                startActivityForResult(takePictureIntent, SELECT_FROM_CAMERA);
            }
        }
    }


    // 建立圖片檔案
    private File createImageFile() throws IOException {
        String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss").format(new Date());  // 檔名以時間命名
        String imageFileName = "JPEG_" + timeStamp + "_";  // 完善檔名
        File storageDir = getExternalFilesDir(Environment.DIRECTORY_PICTURES);  // 獲取檔案儲存路徑
        File image = File.createTempFile(imageFileName, ".jpg", storageDir);
        currentPhotoPath = image.getAbsolutePath();
        return image; // 返回檔案
    }

接下來就是呼叫相簿的程式碼了,前面提到過,呼叫相簿需要動態申請許可權,雖然參考Android官方提供的程式碼不申請也是可以正常開啟的,但是獲取不到圖片的真實路徑(可能是我技術不到位),這裡就申請一次吧

 // 詢問使用者是否授權
    private void selectImageFromAlbum() {
        // 詢問使用者是否授予許可權
        if (ContextCompat.checkSelfPermission(SelectImageActivity.this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {
            ActivityCompat.requestPermissions(SelectImageActivity.this, new String[]{Manifest.permission.READ_EXTERNAL_STORAGE}, 1);  // 請求許可權
        } else {  // 已經授予許可權
            openAlbum();  // 呼叫相簿
        }

    }

    // 判斷使用者是否授予許可權
    @Override
    public void onRequestPermissionsResult(int requestCode, @NonNull @NotNull String[] permissions, @NonNull @NotNull int[] grantResults) {
        if (requestCode == 1) {
            if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {  // 使用者給予許可權
                openAlbum();
            } else {
                Toast.makeText(SelectImageActivity.this, "請授予讀取相簿許可權!", Toast.LENGTH_SHORT).show();
            }
        }
    }

    // 從相簿選取圖片處理
    private String handleImage(Intent data) {
        String imagePath = null;
        Uri uri = data.getData();
        if (DocumentsContract.isDocumentUri(this, uri)) {
            String docId = DocumentsContract.getDocumentId(uri);
            if ("com.android.providers.media.documents".equals(uri.getAuthority())) {
                String id = docId.split(":")[1];
                String selection = MediaStore.Images.Media._ID + "=" + id;
                imagePath = 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.parseLong(docId));
                imagePath = getImagePath(contentUri, null);
            }
        } else if ("content".equalsIgnoreCase(uri.getScheme())) {
            imagePath = getImagePath(uri, null);
        } else if ("file".equalsIgnoreCase(uri.getScheme())) {
            imagePath = uri.getPath();
        }
        return imagePath;  // 使用者選擇圖片的真實地址
    }


    // 獲得相簿中圖片的真實路徑
    private String getImagePath(Uri externalContentUri, String selection) {
        String path = null;
        Cursor cursor = getContentResolver().query(externalContentUri, null, selection, null, null);
        if (cursor != null) {
            if (cursor.moveToNext()) {
                path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA));
            }
            cursor.close();
        }
        return path;
    }

    // 開啟相簿
    private void openAlbum() {
        Intent intent = new Intent("android.intent.action.GET_CONTENT");
        intent.setType("image/*");
        startActivityForResult(intent, SELECT_FROM_ALBUM); // 跳轉意圖
    }

呼叫完相簿和相機後需要返回到 SelectImageActivity進行處理,這裡我們重寫 onActivityResult 方法處理結果,為什麼只需要處理相簿的返回結果呢?因為相機拍攝圖片的路徑我們已經知道了,儲存在 currentPhotoPath 了,所以這裡只處理從相簿中選擇圖片的地址

    //Activity返回結果處理
    @Override
    protected void onActivityResult(int requestCode, int resultCode, @Nullable @org.jetbrains.annotations.Nullable Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        String imagePath = null;
        if (resultCode == RESULT_OK) {  // 返回碼為真
            if (requestCode == SELECT_FROM_ALBUM) {
                assert data != null;
                currentPhotoPath = handleImage(data);
            }
            returnImagePath(currentPhotoPath);
        }else {
            finish();  // 如果返回值不是RESULT_OK, 這種情況可能是使用者放棄選擇圖片,此時直接結束該activity否則會顯示當前activity
        }
    }

現在我們已經能獲取拍攝圖片或者使用者選擇圖片的真實地址了,那麼我們需要把這個地址返回給呼叫它的Activity,程式碼實現如下

 private void returnImagePath(String imgPath) {
        Intent intent = new Intent();
        intent.putExtra("imagePath", imgPath);
        setResult(RESULT_OK, intent);
        finish();
    }

好了,基本完成了,現在我們可以獲取到圖片的真實地址並進行返回,但是還需要知道呼叫的Activity是要開啟相簿的還是開啟相機,因此我們需要接收上一級Activity傳送過來的資訊,程式碼如下

 @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_select_image);

        int imageWhere = getIntent().getIntExtra("imageFromWhere", 0);
        switch (imageWhere) {
            case SELECT_FROM_CAMERA:  // 拍攝新圖片
                selectImageFromCamera();
                break;
            case SELECT_FROM_ALBUM:  // 從相簿中選取
                openAlbum();
                break;
        }
    }

到這裡,我們介面以及設定完畢了,但由於SelectImageActivity是個空白的Activity,跳轉過程會有空白一閃而過,對於“高質量”程式設計師,這是不能忍受的,所以我這裡把SelectImageActivity設定為透明,這樣就不會有空白了,怎麼設定呢?,只需要在AndroidManifest檔案中把當前的Activity後面加上這句

<activity android:name="function.SelectImageActivity" android:theme="@android:style/Theme.Translucent"/>

注意,如果你使用這種方法,SelectImageActivity不能繼承AppCompatActivity,而是向我一樣繼承Activity,否則會報錯,到這裡SelectImageActivity的配置完成,下面我們需要呼叫一下,呼叫SelectImageActivity的Activity中的程式碼如下

 
Intent intent = new Intent(EncryptionActivity.this, SelectImageActivity.class);

case R.id.tv_image_select_from_album: // 從相簿中選取
    intent.putExtra("imageFromWhere", SELECT_FROM_ALBUM);
    startActivityForResult(intent, 1);
    break; case R.id.tv_image_select_from_camera: // 拍攝新圖片
    intent.putExtra("imageFromWhere", SELECT_FROM_CAMERA);
    startActivityForResult(intent,
1);
    break;

ok,這裡我們在需要呼叫的Activity中傳送兩個整形數字,來告訴SelectImageActivity我們是要呼叫相機還是開啟相簿,當SelectImageActivity完成並返回後,我們就能在onActivityResult中接收到圖片的真實路徑了

 @Override
    protected void onActivityResult(int requestCode, int resultCode, @Nullable @org.jetbrains.annotations.Nullable Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if (resultCode == RESULT_OK){
            Bitmap bitmap = BitmapFactory.decodeFile(data.getStringExtra("imagePath"));
            mIvSelectImage.setImageBitmap(bitmap);
        }
    }

這裡,我直接把圖片放在一個Image View中,全部過程完成!

Android開發新人,寫的不好,大神勿噴,一個簡單的案例,希望能夠幫助到你

 

相關文章