如何按下按钮扫描QR码?

时间:2015-12-15 13:53:36

标签: ios swift qr-code camera-view

我使用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并启用扫描。

这里有一个草图:

enter image description here

修改 我的快速修复可能不是正确的方法。

我替换了

 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
        }
        } 

2 个答案:

答案 0 :(得分:2)

这个教程的作者在这里。我的方法是使用专用的扫描视图控制器,但我想你想用你现有的视图控制器统一它 - 这没关系。这两种方法都有效。

如果您想一直显示相机界面(即使没有主动识别QR码),那么您打算使用布尔来跟踪是否启用扫描是一个很好的选择。我的示例代码有 List<string> GroupNames = new List<string>(); GroupNames.AddRange(ADConnect.ADConnect.GetGroups(adusername)); List<string> FilteredGroup = new List<string>(); var FilteredName = new String[] { "Board" }; var names = GroupNames.Where(t => FilteredName.Any(c => t.Contains(c))); FilteredGroup.AddRange(names); cmbgroup.DataSource = FilteredGroup; 方法被调用,并在找到代码时调用foundCode()

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

这样的事情应该这样做:

scanEnabled

如果您愿意,可以将func foundCode(code: String) { if scanCode == true { print(code) captureSession.stopRunning() AudioServicesPlaySystemSound(SystemSoundID(kSystemSoundID_Vibrate)) dismissViewControllerAnimated(true, completion: nil) } } 检查移至scanCode == true以保存不必要的方法调用。

答案 1 :(得分: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)
}