如何在Swift 4中成功扫描二维码后关闭iOS相机

4

我如何在成功扫描QR码后关闭相机?我的问题是,当我从QR码扫描时,相机在扫描后没有关闭。我该如何关闭相机? 我的代码:

var captureSession: AVCaptureSession!
var previewLayer: AVCaptureVideoPreviewLayer!
var qrCode = String()

func failed() {
        let ac = UIAlertController(title: "Scanning not supported", message: "Your device does not support scanning a code from an item. Please use a device with a camera.", preferredStyle: .alert)
        ac.addAction(UIAlertAction(title: "OK", style: .default))
        present(ac, animated: true)
        captureSession = nil
    }

    func metadataOutput(_ output: AVCaptureMetadataOutput, didOutput metadataObjects: [AVMetadataObject], from connection: AVCaptureConnection) {
        captureSession.stopRunning()

        if let metadataObject = metadataObjects.first {
            guard let readableObject = metadataObject as? AVMetadataMachineReadableCodeObject else { return }
            guard let stringValue = readableObject.stringValue else { return }
            AudioServicesPlaySystemSound(SystemSoundID(kSystemSoundID_Vibrate))
            found(code: stringValue)
        }

        dismiss(animated: true)

    }

    func found(code: String) {
        print(code)
        qrCode = code
    }

    override var prefersStatusBarHidden: Bool {
        return true
    }

    override var supportedInterfaceOrientations: UIInterfaceOrientationMask {
        return .portrait
    }

打开相机的按钮:

 @IBAction func scanQrButton(_ sender: Any) {
    view.backgroundColor = UIColor.black
    captureSession = AVCaptureSession()

    guard let videoCaptureDevice = AVCaptureDevice.default(for: .video) else { return }
    let videoInput: AVCaptureDeviceInput

    do {
        videoInput = try AVCaptureDeviceInput(device: videoCaptureDevice)
    } catch {
        return
    }

    if (captureSession.canAddInput(videoInput)) {
        captureSession.addInput(videoInput)
    } else {
        failed()
        return
    }

    let metadataOutput = AVCaptureMetadataOutput()

    if (captureSession.canAddOutput(metadataOutput)) {
        captureSession.addOutput(metadataOutput)

        metadataOutput.setMetadataObjectsDelegate(self, queue: DispatchQueue.main)
        metadataOutput.metadataObjectTypes = [.qr]
    } else {
        failed()
        return
    }

    previewLayer = AVCaptureVideoPreviewLayer(session: captureSession)
    previewLayer.frame = view.layer.bounds
    previewLayer.videoGravity = .resizeAspectFill
    view.layer.addSublayer(previewLayer)

    captureSession.startRunning()
}

正如你所见,我有一个打开相机并扫描QR的代码,但我想在代码扫描后关闭相机而不用转场。

2个回答

2
你尝试过这个吗?
self.captureSession?.stopRunning()

1
previewLayer.removeFromSuperlayer() - yveszenne

2

在执行扫描后,您可以在会话上使用实例方法stopRunning()来停止捕获:

captureSession.stopRunning()

然后使用相机输出的预览界面,关闭视图(我认为可以使用 removeFromSuperlayer() 方法)。 提供的 AVCaptureMetadataOutputObjectsDelegate 可以使用以下方法来检测何时扫描了 QR 码:

optional func metadataOutput(_ output: AVCaptureMetadataOutput, didOutput metadataObjects: [AVMetadataObject], from connection: AVCaptureConnection)

根据苹果文档,您还应将startRunning()方法放在串行队列中:
startRunning()方法是一个阻塞调用,可能需要一些时间,因此,您应该在串行队列上执行会话设置,以避免阻塞主队列(这可以保持UI的响应)。

我不需要停止相机,我想关闭。 - Andrew
你的意思是“从屏幕上移除它”吗? - adrgrondin
我找到了解决方案:self.previewLayer?.isHidden = true self.qrCodeFrameView?.isHidden = true self.captureSession?.stopRunning() - Andrew
2
不要忘记,每次调用scanQrButton()时,您都会在视图层次结构中添加一个新的预览层(它将保留在内存中)。您可以使用view.layer.sublayers = nil来删除该层,或者使用我的答案中的方法进行删除。祝你好运! - adrgrondin

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接