我使用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
}
}
答案 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)
}