2

我正在使用https://www.hackingwithswift.com/example-code/media/how-to-scan-a-qr-code提供的代码来制作我自己的扫描应用程序。但我喜欢在按下按钮时进行扫描。现在为此,我将viewDidLoad()教程中的部分放入它自己的函数中:

func cameraScanningLayer(){
    view.backgroundColor = UIColor.blackColor()
    captureSession = AVCaptureSession()

    let videoCaptureDevice = AVCaptureDevice.defaultDeviceWithMediaType(AVMediaTypeVideo)
    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: dispatch_get_main_queue())
        // need to scan barcode + QRcode
        metadataOutput.metadataObjectTypes = [AVMetadataObjectTypeQRCode,AVMetadataObjectTypeEAN8Code, AVMetadataObjectTypeEAN13Code, AVMetadataObjectTypePDF417Code,AVMetadataObjectTypeCode128Code,AVMetadataObjectTypeCode39Code]
    } else {
        failed()
        return
    }

    // Previewlayer with camera
    previewLayer = AVCaptureVideoPreviewLayer(session: captureSession);
    previewLayer.frame = viewForLayer.bounds;
    previewLayer.videoGravity = AVLayerVideoGravityResizeAspectFill;
    viewForLayer.layer.addSublayer(previewLayer);

    captureSession.startRunning();
}

一个按钮动作调用该函数:

func buttonScanAction() {
    print("Scan")
    scanEnabled = true // like to use some kind of bool/switch
    self.cameraScanningLayer()
}

我遇到的问题是:

1)加载时相机不在视野中

2)按下按钮后,相机在视野中,但它总是自动扫描

所以我想到了使用全局:

var scanEnabled: Bool = false

然后,当单击按钮时,将其设置为 true 并启用扫描。

这里有一个草图供参考:

在此处输入图像描述

编辑 我的快速修复,这可能不是正确的方法。

我更换了

 let metadataOutput = AVCaptureMetadataOutput() {...} else {
            failed()
            return
        }

并将其放在 if 语句之间

if (scanEnabled == true) {

        let metadataOutput = AVCaptureMetadataOutput()

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

            metadataOutput.setMetadataObjectsDelegate(self, queue: dispatch_get_main_queue())
            // to use them both wwe need to skip AVMetadataObjectTypeQRCode
            metadataOutput.metadataObjectTypes = [AVMetadataObjectTypeQRCode,AVMetadataObjectTypeEAN8Code, AVMetadataObjectTypeEAN13Code, AVMetadataObjectTypePDF417Code,AVMetadataObjectTypeCode128Code,AVMetadataObjectTypeCode39Code]
            scanEnabled = false
        } else {
            failed()
            return
        }
        } 
4

2 回答 2

4

该教程的作者在这里。我的方法是使用专用的扫描视图控制器,但我想你想将它与现有的视图控制器统一起来——这很好。两种方法都有效。

如果您想一直显示相机界面(即使没有主动识别 QR 码),那么您计划使用布尔值来跟踪是否启用扫描是一个不错的选择。我的示例代码有一个foundCode()被调用的方法,并且dismissViewControllerAnimated()在找到代码时也会调用。

在您的版本中,您需要foundCode()完成停止扫描、处理解雇等所有工作。然后您可以scanEnabled在一个地方为您的布尔值添加检查。

这样的事情应该这样做:

func foundCode(code: String) {
    if scanCode == true {
        print(code)

        captureSession.stopRunning()
        AudioServicesPlaySystemSound(SystemSoundID(kSystemSoundID_Vibrate))
        dismissViewControllerAnimated(true, completion: nil)
    }
}

如果您愿意,可以将scanCode == true检查上移didOutputMetadataObjects以保存不必要的方法调用。

于 2015-12-15T14:06:35.043 回答
0

感谢@alex 提出这个问题。我还使用了twostraws创建的优秀类(非常有用的Paul,非常感谢),并且还需要仅通过按钮操作来实现扫码阅读。我的解决方案如下:

我将其定义metadataOutput为全局变量,并且仅在按钮操作中将它们集成为委托:

var metadataOutput: AVCaptureMetadataOutput!

viewDidLoad方法:

metadataOutput = AVCaptureMetadataOutput()

if (captureSession.canAddOutput(metadataOutput)) {
    captureSession.addOutput(metadataOutput)
    // Was removed this line: metadataOutput.setMetadataObjectsDelegate(self, queue: DispatchQueue.main)
    metadataOutput.metadataObjectTypes = [.qr]
} else {
    failed()
    return
}

func buttonScanAction() {
    print("Scan")
    metadataOutput.setMetadataObjectsDelegate(self, queue: DispatchQueue.main)
}

当我改变视图时,我会停止摄像头并像这样删除代表:

override func viewWillDisappear(_ animated: Bool) {
    super.viewWillDisappear(animated)
    if (captureSession?.isRunning == true) {
        captureSession.stopRunning()
    }
    metadataOutput.setMetadataObjectsDelegate(nil, queue: DispatchQueue.main)
}
于 2021-03-24T13:07:09.460 回答