【问题标题】:After a barcode scan the screen freezes条码扫描后屏幕冻结
【发布时间】:2018-09-03 15:14:07
【问题描述】:

我正在学习 SWIFT 并变得越来越好。我拥有的条形码应用程序可以工作,但它会无休止地扫描条形码,直到我强制弹出屏幕“确定”、“取消”来接受或拒绝扫描。在研究中,我发现了非常相似的代码(如下所示)。所以我创建了一个只有一个视图控制器的测试项目来尝试这段代码。它工作得很好,但一旦扫描条形码就会冻结。我的意图是取消冻结屏幕,接受并保存它并允许用户扫描更多条码。任何帮助是极大的赞赏。我尝试在找到的行下方停止扫描(代码:),但它无助于解冻。我可能是错的,但我怀疑 viewilldisapper 永远不会在这里被调用。


import AVFoundation
import UIKit

class ScannerViewController: UIViewController, AVCaptureMetadataOutputObjectsDelegate {
var captureSession: AVCaptureSession!
var previewLayer: AVCaptureVideoPreviewLayer!

override func viewDidLoad() {
    super.viewDidLoad()

    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 = [.ean8, .ean13, .pdf417]
    } else {
        failed()
        return
    }

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

    captureSession.startRunning()
}

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
}

override func viewWillAppear(_ animated: Bool) {
    super.viewWillAppear(animated)

    if (captureSession?.isRunning == false) {
        captureSession.startRunning()
    }
}

override func viewWillDisappear(_ animated: Bool) {
    super.viewWillDisappear(animated)

    if (captureSession?.isRunning == true) {
        captureSession.stopRunning()
    }
}

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

override var prefersStatusBarHidden: Bool {
    return true
}

override var supportedInterfaceOrientations: UIInterfaceOrientationMask {
    return .portrait
}

}


修改后的代码:

 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)

        barcodeValue = stringValue
    }

    // dismiss(animated: true)
    previewLayer.removeFromSuperlayer()

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

    let alertController = UIAlertController(title: "Barcode Scanned", message: barcodeValue!, preferredStyle: .alert)
    alertController.addAction(UIAlertAction(title: "OK", style: .default, handler: {(alert: UIAlertAction!) in self.restartScan()}))
    alertController.addAction(UIAlertAction(title: "Cancel", style: .cancel, handler:nil))
    present(alertController, animated: true, completion: nil)
    //captureSession.startRunning()
}

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

func restartScan() {
 captureSession.startRunning()
}

【问题讨论】:

    标签: ios swift barcode freeze scanning


    【解决方案1】:

    这不是冻结,当检测到第一个条形码时,您会在 func metadataOutput(_ output: AVCaptureMetadataOutput 内使用 captureSession.stopRunning() 停止该过程,您需要这个

    previewLayer.removeFromSuperlayer()
    

    而不是这个

    dismiss(animated: true)
    

    因为它是一个层而不是你之前添加的 vc 层

    view.layer.addSublayer(previewLayer)
    

    注意:你的视图背景是黑色的

    【讨论】:

    • 感谢您的及时回复。我修复了代码;这一次,正如您所建议的那样,屏幕变黑,因为它被设置为黑色。我正在寻找激活的相机并再次开始扫描
    • 然后评论这一行 captureSession.stopRunning() ,顺便说一句,您可以将该代码嵌入到 VC 中并放置一个完成按钮以关闭
    • 谢谢。那行得通,但它一直在扫描代码。我只需要一次扫描。看看你的另一个建议,当你“在 VC 内部”时,你是指对新 VC 的继承吗?还是为最终用户弹出警报并关闭警报?
    • 您可以将其设置为模态 vc / view ,在您当前的实现中更改相机层的框架并在其上方添加一个用于切换隐藏/显示的按钮
    • 只扫描一个取消注释此 **captureSession.stopRunning() ** 并按照我告诉你的方式删除图层,第一次扫描后将立即删除扫描视图
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2014-02-06
    • 2020-11-10
    • 1970-01-01
    • 2011-03-03
    • 1970-01-01
    相关资源
    最近更新 更多