Как получить 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();
}
}
Это будет работать во всех версиях. Я тоже это проверял.