Android - намерение камеры приводит к сбою приложения

Я пытаюсь активировать камеру с намерением по следующим кодам:

import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.net.Uri;
import android.os.Bundle;
import android.os.Environment;
import android.provider.MediaStore;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.Toast;
import java.io.File;
import java.io.IOException;
import java.text.SimpleDateFormat;
import java.util.Date;

public class MainActivity extends AppCompatActivity {
    private static final String TAG = "MainActivity";
    private ImageView mImageView;
    static final int REQUEST_IMAGE_CAPTURE = 1;
    static final int REQUEST_TAKE_PHOTO = 1;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Button button = (Button) findViewById(R.id.button);
        mImageView = findViewById(R.id.imageView);

        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Toast.makeText(MainActivity.this, "clicked", Toast.LENGTH_SHORT).show();
                dispatchTakePictureIntent();
            }
        });
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        Log.d(TAG, "onActivityResult: 0");
        if (requestCode == REQUEST_IMAGE_CAPTURE && resultCode == RESULT_OK) {
            Log.d(TAG, "onActivityResult: 1");

            Bundle extras = data.getExtras();
            Bitmap imageBitmap = (Bitmap) extras.get("data");
            mImageView.setImageBitmap(imageBitmap);
        }
    }

    private void dispatchTakePictureIntent() {
        Log.d(TAG, "dispatchTakePictureIntent: 0");
        Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
        if (takePictureIntent.resolveActivity(getPackageManager()) != null) {
            Log.d(TAG, "dispatchTakePictureIntent: 1");
            startActivityForResult(takePictureIntent, REQUEST_IMAGE_CAPTURE);
        }
    }
}

Приложение вылетает каждый раз, когда я нажимаю кнопку (оно должно открывать камеру). Я могу получить «dispatchTakePictureIntent: 1» из logcat, поэтому takePictureIntent.resolveActivity (getPackageManager ()) не возвращает null. Я думаю, что startActivityForResult (takePictureIntent, REQUEST_IMAGE_CAPTURE) вызывает сбой приложения.

Коды дублируются из официальных документов: https://developer.android.com/training/camera/photobasics#java.

Я пробовал dispatchTakePictureIntent (View v), но приложение все равно вылетало. Я также заменил dispatchTakePictureIntent () приведенным ниже методом, и он все равно аварийно завершил работу.

private void TakePicture(){
    Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");
    startActivityForResult(intent, 0);
}

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

Убедитесь, что вы указали разрешение камеры в файле манифеста.

Raj 01.12.2018 11:17

Используйте Logcat, чтобы исследовать трассировку стека Java, связанную с вашим сбоем, вместо того, чтобы делать предположения: stackoverflow.com/q/23353173/115145

CommonsWare 01.12.2018 13:43
1
2
2 067
3

Ответы 3

Я думаю, вы не использовали какую-либо проверку разрешений, потому что для устройства Android ниже 23 не нужно запрашивать разрешение, но выше 23 вам нужно объявить или попросить пользователя получить доступ к функциям Android.

Если устройство работает под управлением Android 6.0 (уровень API 23) или выше, а targetSdkVersion приложения составляет 23 или выше, пользователь не уведомляется о каких-либо разрешениях приложения во время установки. Ваше приложение должно попросить пользователя предоставить опасные разрешения во время выполнения. Когда ваше приложение запрашивает разрешение, пользователь видит системный диалог (как показано на рисунке 1 слева), сообщающий пользователю, к какой группе разрешений ваше приложение пытается получить доступ. В диалоговом окне есть кнопки «Запретить» и «Разрешить».

Согласно этому Док ..

https://developer.android.com/guide/topics/permissions/overview


 if (ContextCompat.checkSelfPermission(thisActivity,    Manifest.permission.WRITE_CALENDAR)
        != PackageManager.PERMISSION_GRANTED) {
    // Permission is not granted
}


    // Here, thisActivity is the current activity
if (ContextCompat.checkSelfPermission(thisActivity,
        Manifest.permission.READ_CONTACTS)
        != PackageManager.PERMISSION_GRANTED) {

    // Permission is not granted
    // Should we show an explanation?
    if (ActivityCompat.shouldShowRequestPermissionRationale(thisActivity,
            Manifest.permission.READ_CONTACTS)) {
        // Show an explanation to the user *asynchronously* -- don't block
        // this thread waiting for the user's response! After the user
        // sees the explanation, try again to request the permission.
    } else {
        // No explanation needed; request the permission
        ActivityCompat.requestPermissions(thisActivity,
                new String[]{Manifest.permission.READ_CONTACTS},
                MY_PERMISSIONS_REQUEST_READ_CONTACTS);

        // MY_PERMISSIONS_REQUEST_READ_CONTACTS is an
        // app-defined int constant. The callback method gets the
        // result of the request.
    }
} else {
    // Permission has already been granted
}

Пример кода из Android-документа https://developer.android.com/training/permissions/requesting#java

У меня есть <uses-permission android: name = "android.permission.CAMERA" /> <uses-permission android: name = "android.permission.WRITE_EXTERNAL_STORAGE" /> в моем Manifest.xml, или мне нужно что-то добавить в добиться проверки разрешения?

Jerry Chen 01.12.2018 10:50

как уже упоминал @KarshSoni, вы должны спросить у пользователя разрешение камеры, а не просто объявить его в своем манифесте. Подробнее об этом можно прочитать здесь: developer.android.com/training/permissions/requesting

Mohammed 01.12.2018 11:25

Пожалуйста, попробуйте дать разрешение камеры в файле манифеста Android. <uses-permission android:name = "android.permission.CAMERA"> </uses-permission>. а затем попробуйте заменить свой код намерения этим Intent intent = new Intent("android.media.action.IMAGE_CAPTURE"); startActivity(intent);

если вы не хотите использовать камеру устройства, вы также можете создать собственную пользовательскую камеру с помощью этой библиотеки https://github.com/natario1/CameraView

вы можете использовать этот способ

public static final int REQUEST_PICTURE_FROM_GALLERY = 10001;
public static final int REQUEST_PICTURE_FROM_CAMERA = 10002;

private File tempFileFromSource = null;
private Uri tempUriFromSource = null;
    
public void selectImageFromGallery() {
        
        try {

            tempFileFromSource  = File.createTempFile("choose", "png", context.getExternalCacheDir());
            tempUriFromSource   = Uri.fromFile(tempFileFromSource);

            Intent intent = new Intent(Intent.ACTION_PICK, android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
            intent.putExtra(MediaStore.EXTRA_OUTPUT, tempUriFromSource);
            intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
            
            context.startActivityForResult(intent, REQUEST_PICTURE_FROM_GALLERY);

        } catch (IOException e) {
            e.printStackTrace();
        }

    }

 
    public void takePhotoWithCamera() {
        
        try {

            tempFileFromSource  = File.createTempFile("choose_", ".png", context.getExternalCacheDir());
            tempUriFromSource = FileProvider.getUriForFile(context, context.getPackageName() + ".fileprovider", tempFileFromSource);

            Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
            intent.putExtra(MediaStore.EXTRA_OUTPUT, tempUriFromSource);
            intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
            intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
            
            context.startActivityForResult(intent, REQUEST_PICTURE_FROM_CAMERA);

        } catch (IOException e) {
            e.printStackTrace();
        }


    }


    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        
        if ((requestCode == REQUEST_PICTURE_FROM_GALLERY) && (resultCode == Activity.RESULT_OK)) {
            Log.d(TAG, "Image selected from gallery");
            prepareImage(data.getData(), tempFileFromSource);
        } else if ((requestCode == REQUEST_PICTURE_FROM_CAMERA) && (resultCode == Activity.RESULT_OK)) {
            Log.d(TAG, "Image selected from camera");
            prepareImage(tempUriFromSource, tempFileFromSource);
        }
    }

    private void prepareImage(Uri uri, File imageFile) {
        try {

            if (uri == null)
                return;

            Bitmap bitmap = BitmapUtils.decodeBitmap(context, uri, 300, 300);

            if (bitmap == null)
                return;
            
            // now you have bitmap
        } catch (IOException e) {
            e.printStackTrace();
        }
    }


    public static Bitmap decodeBitmap(Context context, Uri uri, int maxWidth, int maxHeight) throws IOException {

        // First decode with inJustDecodeBounds=true to check dimensions
        final BitmapFactory.Options options = new BitmapFactory.Options();
        options.inJustDecodeBounds = true;
        InputStream imageStream = context.getContentResolver().openInputStream(uri);
        BitmapFactory.decodeStream(imageStream, null, options);
        if (imageStream != null)
            imageStream.close();

        // Calculate inSampleSize
        options.inSampleSize = calculateSize(options, maxWidth, maxHeight);

        // Decode bitmap with inSampleSize set
        options.inJustDecodeBounds = false;
        imageStream = context.getContentResolver().openInputStream(uri);
        Bitmap img = BitmapFactory.decodeStream(imageStream, null, options);

        img = rotateImageIfRequired(context, img, uri);
        return img;
    }

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