安装后相机无法在第一个应用程序 运行 上运行

Camera not working on first app run after install

我在我的 Android 应用程序中实现了一个简单的二维码扫描器代码。该代码似乎工作正常,但由于某种原因,在安装后,您第一次打开该应用程序时,CameraView 保持黑色。只有在我第二次启动该应用程序后它才能工作(关闭后,onResume 不会改变任何东西)

这是我的代码:

private static Context context;

private CameraSource cameraSource;
private SurfaceView cameraView;
private final int MY_PERMISSIONS_REQUEST_CAMERA = 1;
private String token = "";
private String tokenanterior = "";
private static Activity mActivity;

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

    // verifico si el usuario dio los permisos para la camara
    if (ActivityCompat.checkSelfPermission(MainActivity.this, Manifest.permission.CAMERA)
            != PackageManager.PERMISSION_GRANTED) {

        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            // verificamos la version de ANdroid que sea al menos la M para mostrar
            // el dialog de la solicitud de la camara
            if (shouldShowRequestPermissionRationale(
                    Manifest.permission.CAMERA)) ;
            requestPermissions(new String[]{Manifest.permission.CAMERA},
                    MY_PERMISSIONS_REQUEST_CAMERA);
        }
    }

    cameraView = (SurfaceView) findViewById(R.id.camera_view);
    initQR();

}

public void initQR() {

    // creo el detector qr
    BarcodeDetector barcodeDetector =
            new BarcodeDetector.Builder(this)
                    .setBarcodeFormats(Barcode.ALL_FORMATS)
                    .build();

    // creo la camara
    cameraSource = new CameraSource
            .Builder(this, barcodeDetector)
            .setRequestedPreviewSize(1600, 1024)
            .setAutoFocusEnabled(true) //you should add this feature
            .build();

    // listener de ciclo de vida de la camara
    cameraView.getHolder().addCallback(new SurfaceHolder.Callback() {
        @Override
        public void surfaceCreated(SurfaceHolder holder) {

            // verifico si el usuario dio los permisos para la camara
            if (ActivityCompat.checkSelfPermission(MainActivity.this, Manifest.permission.CAMERA)
                    != PackageManager.PERMISSION_GRANTED) {

                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
                    // verificamos la version de ANdroid que sea al menos la M para mostrar
                    // el dialog de la solicitud de la camara
                    if (shouldShowRequestPermissionRationale(
                            Manifest.permission.CAMERA)) ;
                    requestPermissions(new String[]{Manifest.permission.CAMERA},
                            MY_PERMISSIONS_REQUEST_CAMERA);
                }
                return;
            }else {
                try {
                    cameraSource.start(cameraView.getHolder());
                } catch (IOException ie) {
                    Log.e("CAMERA SOURCE", ie.getMessage());
                }
            }
        }

        @Override
        public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
        }

        @Override
        public void surfaceDestroyed(SurfaceHolder holder) {
            cameraSource.stop();
        }
    });

    // preparo el detector de QR
    barcodeDetector.setProcessor(new Detector.Processor<Barcode>() {
        @Override
        public void release() {
        }
}

有什么想法吗?

编辑:根据@r2rek 的回答,我在 class 中实现了 onRequestPermissionResult() 函数,如下所示:

@Override
public void onRequestPermissionsResult(int requestCode,
                                       String[] permissions, int[] grantResults) {
    switch (requestCode) {
        case MY_PERMISSIONS_REQUEST_CAMERA: {
            // If request is cancelled, the result arrays are empty.
            if (grantResults.length > 0
                    && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
                // permission was granted, yay! Do the
                // contacts-related task you need to do.
                Log.i("Results:", "HE GAVE EM");
                initQR();
            } else {
                // permission denied, boo! Disable the
                // functionality that depends on this permission.
            }
            return;
        }

    }
}

但这并没有解决问题,initQR 被执行了两次(因为我被要求获得拍摄视频和拍照的权限)。但是,在第一次安装时,它仍然无法在 SurfaceView 上正确显示相机流。

您是否在清单中实施了许可请求?将它们放在 运行 时间的代码旁边很重要:

<uses-feature android:name="android.hardware.camera" />
<uses-permission android:name="android.permission.CAMERA" />

很简单。您第一次请求许可,但没有对用户允许做出反应。第二次,权限已经被授予。您需要根据以下文档实施 onRequestPermissionResult:

https://developer.android.com/training/permissions/requesting#handle-response

基本上,当用户允许你使用相机时,你需要initQr()

唯一的解决方案最终是在单独的 activity 中请求权限。 因为我希望相机 activity 成为第一个,所以我不得不实现一个新的空白 activity,完全是为了请求权限然后继续相机 activity。

在这里遇到同样的问题。 问题是当您在权限对话框后通过 initQR() 重新创建扫描仪时,不会再次调用 surfaceCreated 回调。所以 cameraSource.start 没有被调用,你面对黑屏。

在其他情况下也会出现同样的问题 - 例如当您通过 USB 数据线将 phone 连接到笔记本电脑时,会弹出安全对话框并削弱相机源。

观察 - 在这些情况下总是调用 onPause()onResume()。有时扫描仪会恢复(翻转后,睡眠),有时不会。

我的解决方案(hack)是将延迟恢复逻辑放入 onPause()onResume() 方法:

@Override
protected void onPause() {
    super.onPause();
    cameraSource.release();
    cameraSourceStarted = false;
}

@Override
protected void onResume() {
    super.onResume();
    initialiseDetectorsAndSources();

    final Handler handler = new Handler(Looper.getMainLooper());
    handler.postDelayed(new Runnable() {
        @Override
        public void run() {
            if (!cameraSourceStarted) {
                try {
                    if (ActivityCompat.checkSelfPermission(MainActivity.this, Manifest.permission.CAMERA) == PackageManager.PERMISSION_GRANTED) {
                        cameraSource.start(surfaceView.getHolder());
                    } else {
                        ActivityCompat.requestPermissions(MainActivity.this, new
                                String[]{Manifest.permission.CAMERA}, REQUEST_CAMERA_PERMISSION);
                    }

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

我还在 surfaceCreated 中设置了我的布尔变量 cameraSourceStarted = true,当 cameraSource 正常启动时 - 所以恢复什么都不做。