Подтвердить что ты не робот

Android-камера: onActivityResult() намерение имеет значение null, если у него есть дополнительные функции

После многого поиска во всех связанных с ним проблемах при переполнении стека и ничего не обнаружив, попробуйте помочь мне.

Я создал намерение захватить изображение. Затем я увидел другое поведение в onActivityResult(): если я не добавлю лишнего в Intent (для небольших фотографий), Intent в onActivityResult в порядке, но когда я добавляю дополнительные функции в намерение писать pic в файл, Цель в onActivityResult null!

Создание намерения:

Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
// without the following line the intent is ok
takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(f));
startActivityForResult(takePictureIntent, actionCode);

Почему это null, и как я могу его решить?

4b9b3361

Ответ 1

То же самое происходит со мной, если вы предоставляете MediaStore.EXTRA_OUTPUT, то намерение равно null, но у вас будет фотография в файле, который вы предоставили (Uri.fromFile(f)).

Если вы не укажете MediaStore.EXTRA_OUTPUT, тогда у вас будет намерение, которое содержит uri из файла, в котором камера сохранила фотографию.

Не знаю, является ли это ошибкой, но работает так.

РЕДАКТИРОВАТЬ: Итак, в onActivityResult() вам больше не нужно проверять данные, если null. Следующие со мной работали:

 @Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    switch (requestCode) {
        case PICK_IMAGE_REQUEST://actionCode
            if (resultCode == RESULT_OK && data != null && data.getData() != null) {
                //For Image Gallery
            }
            return;

        case CAPTURE_IMAGE_REQUEST://actionCode
            if (resultCode == RESULT_OK) {
                //For CAMERA
                //You can use image PATH that you already created its file by the intent that launched the CAMERA (MediaStore.EXTRA_OUTPUT)
                return;
            }
    }
}

Надеюсь, что это поможет

Ответ 2

Образец, написанный в Котлине. Вы создаете приложение Uri для камеры, CameraFragment удерживает его до тех пор, пока камера не вернется к сохранению вашего изображения и не вернет его вам в onActivityResult, как вы ожидали.

CameraFragment.kt

Действует как посредник между приложением для потребителей и камеры. Принимает Uri как ввод и возвращает его в данных Intent.

class CameraFragment : Fragment() {

    companion object {
        val TAG = CameraFragment::class.java.simpleName

        private val KEY_URI = ".URI"

        fun newInstance(uri: Uri, targetFragment: Fragment, requestCode: Int): CameraFragment {
            val args = Bundle()
            args.putParcelable(KEY_URI, uri)
            val fragment = CameraFragment()
            fragment.arguments = args
            fragment.setTargetFragment(targetFragment, requestCode)
            return fragment
        }
    }

    private lateinit var uri: Uri

    private var fired = false

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        retainInstance = true

        fired = savedInstanceState?.getBoolean("fired") ?: false

        if (!fired) {
            val args = arguments
            uri = args.getParcelable(KEY_URI)

            val i = Intent(MediaStore.ACTION_IMAGE_CAPTURE)
            i.putExtra(MediaStore.EXTRA_OUTPUT, uri)
            i.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION)
            context.grantUriPermission(i, uri, Intent.FLAG_GRANT_WRITE_URI_PERMISSION)

            startActivityForResult(i, targetRequestCode)
            fired = true
        }
    }

    override fun onSaveInstanceState(outState: Bundle) {
        super.onSaveInstanceState(outState)
        outState.putBoolean("fired", fired)
    }

    override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
        super.onActivityResult(requestCode, resultCode, data)

        if (requestCode == targetRequestCode) {
            context.revokeUriPermission(uri, Intent.FLAG_GRANT_WRITE_URI_PERMISSION)

            val newData = Intent()
            newData.data = uri
            targetFragment.onActivityResult(requestCode, resultCode, newData)

            dismiss()
        }
    }

    private fun dismiss() {
        fragmentManager.beginTransaction().remove(this).commit()
    }
}

/** Grant Uri permissions for all camera apps. */
fun Context.grantUriPermission(intent: Intent, uri: Uri, modeFlags: Int) {
    val resolvedIntentActivities = packageManager.queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY);
    for (resolvedIntentInfo in resolvedIntentActivities) {
        val packageName = resolvedIntentInfo.activityInfo.packageName;
        grantUriPermission(packageName, uri, modeFlags);
    }
}

Вызвать намерение камеры

this - это фрагмент в вашем приложении, который будет запускать камеру. RC_CAMERA - ваш код запроса для этого действия.

val uri = /* Your output Uri. */
val f = CameraFragment.newInstance(uri, this, RC_CAMERA)
fragmentManager.beginTransaction().add(f, CameraFragment.TAG).commit()

Обработка результата камеры

override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
    super.onActivityResult(requestCode, resultCode, data)

    when(requestCode) {
        RC_CAMERA -> {
            if (resultCode == Activity.RESULT_OK) {
                val uri = data?.data
                // Do whatever you need.
            }
        }
    }
}

Ответ 3

Где вы создали f для Uri.fromFile(f)?

Он должен быть допустимым File объектом. Попробуйте создать его перед строкой EXTRA_OUTPUT.

File f = new File("valid path");

Попробуйте что-то вроде этого:

File file = new File(dataFile);
Uri outFileUri = Uri.fromFile(file);
Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");
intent.putExtra(MediaStore.EXTRA_OUTPUT, outFileUri);
startActivityForResult(intent, TAKE_PHOTO);

Ответ 4

используйте следующее:

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

он работает.