Приложение неожиданно вылетает после съемки, поворачивает экран и возвращается к активности

Приложение неожиданно вылетает после съемки, поворачивает экран и возвращается к активности. Я делаю приложение для Android, которое должно сделать снимок и сохранить маршрут в БД в виде String. Для этого я следовал официальному учебнику Google ( https://developer.android.com/training/camera/photobasics) без реализации моего собственного приложения камеры. Но у меня возникает проблема, когда я выполняю следующие шаги: • Откройте действие, содержащее логику для съемки фотографии, с помощью функции dispatchTakePictureIntent(). • Запустите камеру в портретном режиме. • Поверните экран в горизонтальное положение / камеру в ландшафтный режим. • Сделайте фотографию и коснитесь кнопки "ОК". • Приложение вылетает.

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

Мой код такой:

   override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
        if(requestCode == CodigosDeSolicitud.ANADIR_FOTOGRAFIA && resultCode == Activity.RESULT_OK){
            setPic()
        }
    }

    @Throws(IOException::class)
    private fun createImageFile() : File{

        val timeStamp: String = SimpleDateFormat("yyyyMMdd_HHmmss").format(Date())
        val storageDir : File = getExternalFilesDir(Environment.DIRECTORY_PICTURES)
        return File.createTempFile(
                "JPEG_${timeStamp}_",
                ".jpg",
                storageDir
        ).apply {
            mCurrentPhotoPath = absolutePath
        }

    }

    private fun dispatchTakePicktureIntent(){
        Intent(MediaStore.ACTION_IMAGE_CAPTURE).also {takePictureIntent ->
            takePictureIntent.resolveActivity(packageManager)?.also {
                val photoFile : File? = try{
                    createImageFile()
                } catch (ex : IOException){
                    null
                }
                photoFile?.also {
                    val photoURI: Uri = FileProvider.getUriForFile(
                            this,
                            "com.kps.spart.android.fileprovider",
                            it
                    )
                    takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoURI)
                    startActivityForResult(takePictureIntent, CodigosDeSolicitud.ANADIR_FOTOGRAFIA)
                }
            }

        }
    }

    private fun setPic(){
        val targetW: Int = iconoUsuarioIV.width
        val targetH: Int = iconoUsuarioIV.height

        val bmOptions = BitmapFactory.Options().apply {
            inJustDecodeBounds = true
            BitmapFactory.decodeFile(mCurrentPhotoPath,this)
            val photoW: Int = outWidth
            val photoH: Int = outHeight

            val scaleFactor : Int = Math.min(photoW / targetW, photoH / targetH)

            inJustDecodeBounds = false
            inSampleSize = scaleFactor
            inPurgeable = true
        }

        val exif = ExifInterface(mCurrentPhotoPath)
        val orientation = exif.getAttributeInt(ExifInterface.TAG_ORIENTATION,ExifInterface.ORIENTATION_UNDEFINED)
        Toast.makeText(this@RegistrarUsuarioActivity,"Orientacion: " + orientation, Toast.LENGTH_SHORT).show()

        BitmapFactory.decodeFile(mCurrentPhotoPath, bmOptions)?.also { bitmap ->
            iconoUsuarioIV.setImageBitmap(bitmap)
        }

И мой logcat дает мне следующие ошибки:

Caused by: java.lang.RuntimeException: Failure delivering result ResultInfo{who=null, request=1000, result=-1, data=null} to activity {com.kps.spart.moskimedicationreminder/com.kps.spart.moskimedicationreminder.RegistrarUsuarioActivity}: java.lang.ArithmeticException: divide by zero
    at android.app.ActivityThread.deliverResults(ActivityThread.java:4519)
    at android.app.ActivityThread.performResumeActivity(ActivityThread.java:3777)
    at android.app.ActivityThread.handleResumeActivity(ActivityThread.java:3845) 
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3065) 
    at android.app.ActivityThread.handleRelaunchActivity(ActivityThread.java:4950) 
    at android.app.ActivityThread.-wrap19(Unknown Source:0) 
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1730) 
    at android.os.Handler.dispatchMessage(Handler.java:106) 
    at android.os.Looper.loop(Looper.java:164) 
    at android.app.ActivityThread.main(ActivityThread.java:7000) 
    at java.lang.reflect.Method.invoke(Native Method) 
    at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:441) 
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1408) 
 Caused by: java.lang.ArithmeticException: divide by zero
    at com.kps.spart.moskimedicationreminder.RegistrarUsuarioActivity.setPic(RegistrarUsuarioActivity.kt:223)
    at com.kps.spart.moskimedicationreminder.RegistrarUsuarioActivity.onActivityResult(RegistrarUsuarioActivity.kt:170)
    at android.app.Activity.dispatchActivityResult(Activity.java:7630)
    at android.app.ActivityThread.deliverResults(ActivityThread.java:4515)
    at android.app.ActivityThread.performResumeActivity(ActivityThread.java:3777) 
    at android.app.ActivityThread.handleResumeActivity(ActivityThread.java:3845) 
    at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:3065) 
    at android.app.ActivityThread.handleRelaunchActivity(ActivityThread.java:4950) 
    at android.app.ActivityThread.-wrap19(Unknown Source:0) 
    at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1730) 
    at android.os.Handler.dispatchMessage(Handler.java:106) 
    at android.os.Looper.loop(Looper.java:164) 
    at android.app.ActivityThread.main(ActivityThread.java:7000) 
    at java.lang.reflect.Method.invoke(Native Method) 
    at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:441) 
    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1408) 

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

0 ответов

Я сам наткнулся на это (следуя тому же руководству). Я знаю, что вопрос старый, но я публикую решение для любого другого, с которым он может столкнуться.

Проблема в onCreateметод, который вызывается дважды (когда вы возвращаетесь к активности после съемки, после поворота устройства). Во избежание этого следует предотвратить второй вызовdispatchTakePicktureIntent (что, я полагаю, вы делаете в своем onCreate метод).

Я реализовал это так:

class TakePhotoActivity : AppCompatActivity() {

    private val captureRequestCode = 1
    private val captureState = "capture_in_progress"
    private val capturePhotoPath = "capture_photo_path"
    private var photoPath: String? = null
    private var captureInProgress: Boolean = false

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_take_photo)
        captureInProgress = savedInstanceState?.getBoolean(captureState) ?: false
        photoPath = savedInstanceState?.getString(capturePhotoPath)

        if(!captureInProgress){
            captureInProgress = true
            dispatchTakePictureIntent()
        }
    }

    override fun onSaveInstanceState(outState: Bundle) {
        outState.putBoolean(captureState, captureInProgress)
        outState.putString(capturePhotoPath, photoPath)
        super.onSaveInstanceState(outState)
    }

    override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
        super.onActivityResult(requestCode, resultCode, data)
        if (requestCode == captureRequestCode && resultCode == RESULT_OK) {
            user_photo.adjustViewBounds = true
            val imgFile = File(photoPath ?: "")

            if (imgFile.exists()) {
                val myBitmap = BitmapFactory.decodeFile(imgFile.absolutePath)
                user_photo.setImageBitmap(myBitmap)
                runImageLabelingOnCurrentPicture(myBitmap)
            }
            captureInProgress = false
        }
    }
}

Вам необходимо сохранить флаг "Идет съемка" в savedInstanceState и восстановить его, если он существует в onCreateметод. Это все.

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