Как получить URI захваченного фото?

Чего я хочу достичь? Я хочу получить URI захваченного изображения и сохранить его в Firebase. Что я пробовал? Прежде всего мне нужно было открыть камеру. Ниже как я это сделал:

 Intent cameraIntent = new Intent(android.provider.MediaStore.ACTION_IMAGE_CAPTURE);
    if (cameraIntent.resolveActivity(getActivity().getPackageManager()) != null)
        startActivityForResult(cameraIntent, CAMERA_REQUEST_CODE);

После захвата изображения мне нужно было получить URI изображения. Ниже как я это сделал:

if (resultCode == Activity.RESULT_OK) {
        if (requestCode == CAMERA_REQUEST_CODE) {
            if (data != null) {
                if (data.getData() != null) {
                   Uri capturedImageUri = data.getData(); 
                } else {
                    Toast.makeText(getContext(), getString(R.string.coudldnt_get_photo), Toast.LENGTH_SHORT).show();
                }
            } else {
                Toast.makeText(getContext(), getString(R.string.coudldnt_get_photo), Toast.LENGTH_SHORT).show();
            }

Так что все было в порядке. Но только на некоторых устройствах. Я предполагаю, что он работает только на уровне API более 21 устройства. В других устройствах getData() возвращает ноль.

Итак, что я сделал дальше? Я обнаружил, что могу получить растровое изображение с помощью следующего кода:

Bitmap bitmap = (Bitmap)data.getExtras().get("data") ; 

Итак, у меня есть растровое изображение. И мне нужно было получить URI этого изображения. Ниже как я это сделал:

public Uri getImageUri(Context inContext, Bitmap inImage) {
  String path = 
  Images.Media.insertImage(inContext.getContentResolver(), inImage, 
  "Title", null);
  return Uri.parse(path);
 } 

Выше код возвращает мне URI, но изображение действительно плохого качества.

Итак, я продолжал искать решение. И обнаружил, что правильным способом является создание файла, когда я запускаю CameraActivity и сохраните URI этого файла. А также onActivityResult Я пытался получить этот URI. Ниже как я это сделал:

Intent cameraIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

    if (cameraIntent.resolveActivity(getActivity().getPackageManager()) != null) {
        File photoFile = null;
        try {
            photoFile = createImageFile();
        } catch (IOException ex) {

        }
        if (photoFile != null) {
            Uri photoUri = FileProvider.getUriForFile(getContext(), "i_dont_know_what_to_write_here", photoFile);
            cameraIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoUri);
            startActivityForResult(cameraIntent, CAMERA_REQUEST_CODE);
        }

    }

И мой onActivityResult стали:

    if (resultCode == Activity.RESULT_OK) {
    if (requestCode == CAMERA_REQUEST_CODE) {
        if (data != null) {
 Uri capturedPhotoUri = 
 data.getParcelableExtra(MediaStore.EXTRA_OUTPUT);
   } else {
   Toast.makeText(getContext(), 
   getString(R.string.coudldnt_get_photo), Toast.LENGTH_SHORT).show();
            }

Приведенная выше концепция требует добавления некоторых тегов в манифест:

 <provider
        android:name="android.support.v4.content.FileProvider"
        android:authorities="i_dont_know_what_to_write_here"
        android:exported="false"
        android:grantUriPermissions="true">
        <meta-data
            android:name="android.support.FILE_PROVIDER_PATHS"
            android:resource="@xml/my_paths" />
        </provider>

Мой XML-файл:

<?xml version="1.0" encoding="utf-8"?>
<paths>
<external-files-path name="images"/>
</paths>

Вышеуказанный метод не работает, потому что data на onActivityReuslt был нулевым

Итак, что мне нужно сделать, чтобы достичь своей цели?

2 ответа

Решение

Выполните следующие шаги.

Шаг 1. Создайте файл provider_paths.xml в папке res/xml и напишите в нем код ниже.

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <external-path
        name="external_files"
        path="." />
</paths>

Шаг 2: Объявите провайдера в манифесте, как показано ниже.

       <provider
            android:name="android.support.v4.content.FileProvider"
            android:authorities="${applicationId}.provider"
            android:exported="false"
            android:grantUriPermissions="true">
            <meta-data
                android:name="android.support.FILE_PROVIDER_PATHS"
                android:resource="@xml/provider_paths" />
        </provider>

Шаг - 3: Напишите намерения вашей камеры, как показано ниже.

    Intent m_intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    File file = new File(Environment.getExternalStorageDirectory(), "MyPhoto.jpg");
    Uri uri = FileProvider.getUriForFile(this, this.getApplicationContext().getPackageName() + ".provider", file);
    m_intent.putExtra(android.provider.MediaStore.EXTRA_OUTPUT, uri);
    startActivityForResult(m_intent, REQUEST_CAMERA_IMAGE);

Шаг 4: обработать результат камеры в onActivityResult, как показано ниже.

protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);
    switch (requestCode) {

        //TODO... onCamera Picker Result
        case REQUEST_CAMERA_IMAGE:
            if (resultCode == RESULT_OK) {

              //File object of camera image
                File file = new File(Environment.getExternalStorageDirectory(), "MyPhoto.jpg");

              //Uri of camera image
                Uri uri = FileProvider.getUriForFile(this, this.getApplicationContext().getPackageName() + ".provider", file);

            }
            break;
    }
}

Во время вызова камеры намерения

Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);


 File mediaStorageDir = new File(Environment.getExternalStoragePublicDirectory(
                Environment.DIRECTORY_PICTURES), "IMG_FOLDER");

        try {
            if (!mediaStorageDir.exists()) {
                if (!mediaStorageDir.mkdirs()) {
                    return null;
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }


        imageURI = Uri.fromFile(new File(mediaStorageDir.getPath() + File.separator +
                "profile_img.jpg"));
        intent.putExtra(MediaStore.EXTRA_OUTPUT, imageURI);

        startActivityForResult(intent, Utils.CAMERA_REQUEST);

Внутри onActivityResult

@Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        try {
            if (resultCode == RESULT_OK) {
                if (requestCode == Utils.CAMERA_REQUEST) {
                    if (imageURI != null) {             // use the same uri, that you initialized while calling camera intent
                     // do whatever you want to do with this Uri
                 }
                }
            }
         } catch(Exception E) {}  
  }

Для Nougat, вам нужно добавить этот код ниже в активности всплеск / запуск

if (Build.VERSION.SDK_INT >= 24) {
        try {
            Method m = StrictMode.class.getMethod("disableDeathOnFileUriExposure");
            m.invoke(null);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

Это будет работать во всех версиях. Я тоже это проверял.

Другие вопросы по тегам