Проблемы с доступом к камере с targetSdkVersion 25?

У меня есть приложение, которое использует камеру, она прекрасно работает, когда я скомпилирую targetSdkVersion 23, но когда я пытаюсь использовать version 25 Я получаю эту ошибку:

android.os.FileUriExposedException: файл:///storage/emulated/0/DCIM/IMG_1093948364.jpg, доступный за пределами приложения через ClipData.Item.getUri()

Это код, который я использую:

private void showCameraAction() {
    if(ContextCompat.checkSelfPermission(getContext(), Manifest.permission.WRITE_EXTERNAL_STORAGE)
            != PackageManager.PERMISSION_GRANTED){
        requestPermission(Manifest.permission.WRITE_EXTERNAL_STORAGE,
                getString(R.string.mis_permission_rationale_write_storage),
                REQUEST_STORAGE_WRITE_ACCESS_PERMISSION);
    }else {
        Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        if (intent.resolveActivity(getActivity().getPackageManager()) != null) {
            try {
                mTmpFile = FileUtils.createTmpFile(getActivity());
            } catch (IOException e) {
                e.printStackTrace();
            }
            if (mTmpFile != null && mTmpFile.exists()) {
                intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(mTmpFile));
                startActivityForResult(intent, REQUEST_CAMERA);
            } else {
                Toast.makeText(getActivity(), R.string.mis_error_image_not_exist, Toast.LENGTH_SHORT).show();
            }
        } else {
            Toast.makeText(getActivity(), R.string.mis_msg_no_camera, Toast.LENGTH_SHORT).show();
        }
    }
}

private void requestPermission(final String permission, String rationale, final int requestCode){
    if(shouldShowRequestPermissionRationale(permission)){
        new AlertDialog.Builder(getContext())
                .setTitle(R.string.mis_permission_dialog_title)
                .setMessage(rationale)
                .setPositiveButton(R.string.mis_permission_dialog_ok, new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        requestPermissions(new String[]{permission}, requestCode);
                    }
                })
                .setNegativeButton(R.string.mis_permission_dialog_cancel, null)
                .create().show();
    }else{
        requestPermissions(new String[]{permission}, requestCode);
    }
}

@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
    if(requestCode == REQUEST_STORAGE_WRITE_ACCESS_PERMISSION){
        if(grantResults[0] == PackageManager.PERMISSION_GRANTED){
            showCameraAction();
        }
    } else {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    }
}

Что я должен делать? Благодарю.

2 ответа

Решение

Ваш FileUtils.createTmpFile(getActivity()); вероятно, использует file:// URI, чтобы поделиться файлом с другим приложением (в вашем случае камера).

Версии Android старше 24 используют взамен content:// URI и выдают это исключение, когда вы пытаетесь предоставить общий доступ к файлу напрямую, используя file:// URI.

URI контента позволяет вам предоставлять права на чтение и запись, используя временные права доступа.

Посмотрите на FileProvider.

РЕШЕНИЕ:

Изменено это:

            intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(mTmpFile));

К этому:

            //getActivity() because its a fragment
            Uri uri = FileProvider.getUriForFile(getActivity(),
                    getActivity().getPackageName()
                    , mTmpFile);

            intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);

И это сработало.

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