수색…


소개

QR (빠른 응답) 코드는 기계로 읽을 수있는 광학 레이블에 널리 사용되는 2 차원 바코드입니다. iOS는 iOS 7 이후의 AVFoundation 프레임 워크를 사용하여 QR 코드를 읽을 수있는 방법을 제공합니다. 이 프레임 워크는 카메라를 설정 / 열기하고 카메라 피드에서 QR 코드를 읽을 수있는 API 세트를 제공합니다.

QR을 검색하고 비디오 입력을 표시하는 UIViewController

import AVFoundation
class QRScannerViewController: UIViewController,
     AVCaptureMetadataOutputObjectsDelegate {
   
    func viewDidLoad() {
        self.initCaptureSession()
    }
    
    private func initCaptureSession() {
        let captureDevice = AVCaptureDevice
            .defaultDevice(withMediaType: AVMediaTypeVideo)
        do {
            let input = try AVCaptureDeviceInput(device: captureDevice)
            let captureMetadataOutput = AVCaptureMetadataOutput()
            self.captureSession?.addOutput(captureMetadataOutput)
            captureMetadataOutput.setMetadataObjectsDelegate(self,
                 queue: DispatchQueue.main)
            captureMetadataOutput
                .metadataObjectTypes = [AVMetadataObjectTypeQRCode]
            
            self.videoPreviewLayer = 
                AVCaptureVideoPreviewLayer(session: self.captureSession)
            self.videoPreviewLayer?
                .videoGravity = AVLayerVideoGravityResizeAspectFill
            self.videoPreviewLayer?.frame =    
                self.view.layer.bounds

            self._viewController?.view.layer
                .addSublayer(videoPreviewLayer!)
            self.captureSession?.startRunning()
        } catch {
            //TODO: handle input open error
        }
    }
    private func dismissCaptureSession() {
        if let running = self.captureSession?.isRunning, running {
            self.captureSession?.stopRunning()
        }
        self.captureSession = nil
        self.videoPreviewLayer?.removeFromSuperLayer()
        self.videoPreviewLayer = nil
    }
    
    func captureOutput(_ captureOutput: AVCaptureOutput, 
        didOutputMetadataObjects metadataObjects: [Any]!, 
        from connection: AVCaptureConnection) {
        guard metadataObjects != nil && metadataObjects.count != 0 else {
            //Nothing captured
            return
        }

        if let metadataObj = 
            metadataObjects[0] as? AVMetadataMachineReadableCodeObject {
            guard metadataObj.type == AVMetadataObjectTypeQRCode else {
                return
            }

            let barCodeObject = videoPreviewLayer?
                .transformedMetadataObject(for: 
                    metadataObj as AVMetadataMachineReadableCodeObject)
                 as! AVMetadataMachineReadableCodeObject
        
            if let qrValue = metadataObj.stringValue {
                self.handleQRRead(value: qrValue)
            }
        }
    }

    private handleQRRead(value: String) {
        //TODO: Handle the read qr
    }
    private captureSession: AVCaptureSession?
    private videoPreviewLayer: AVCaptureVideo
}

handleQRRead - 성공적인 스캔에 호출됩니다 initCaptureSession - QR 카메라 입력에 대한 스캔을 초기화 dismissCaptureSession - 카메라 입력을 숨기고 검색을 중지

AVFoudation 프레임 워크로 QR 코드 스캔하기

iOS 7 이전에는 QR 코드를 스캔 할 때 zBar 또는 zXing 과 같은 타사 프레임 워크 또는 라이브러리에 의존해야했습니다. 그러나 Apple은 iOS 7에서 바코드 판독을 위해 AVCaptureMetaDataOutput 을 도입 AVCaptureMetaDataOutput .

AVFoundation 을 사용하여 QR 코드를 읽으려면 AVCaptureSession 을 설정 / 생성하고 captureOutput:didOutputMetadataObjects:fromConnection: delegate 메서드를 captureOutput:didOutputMetadataObjects:fromConnection: 합니다.

1 단계

AVFoundation 프레임 워크 가져 오기 및 AVCaptureMetadataOutputObjectsDelegate 프로토콜 확인

import AVFoundation
class ViewController: UIViewController, AVCaptureMetadataOutputObjectsDelegate

2 단계

QR 코드 판독은 전적으로 비디오 캡처를 기반으로합니다. 따라서 연속 비디오를 캡처하려면 AVCaptureSession 만들고 장치 입력 및 출력을 설정하십시오. 뷰 컨트롤러 viewDidLoad 메서드에 아래 코드를 추가하십시오.

// Create an instance of the AVCaptureDevice and provide the video as the media type parameter.
let captureDevice = AVCaptureDevice.defaultDevice(withMediaType: AVMediaTypeVideo)
 
do {
    // Create an instance of the AVCaptureDeviceInput class using the device object and intialise capture session
    let input = try AVCaptureDeviceInput(device: captureDevice)
    captureSession = AVCaptureSession()
    captureSession?.addInput(input)
    
    // Create a instance of AVCaptureMetadataOutput object and set it as the output device the capture session.
    let captureMetadataOutput = AVCaptureMetadataOutput()
    captureSession?.addOutput(captureMetadataOutput)
    // Set delegate with a default dispatch queue
    captureMetadataOutput.setMetadataObjectsDelegate(self, queue: DispatchQueue.main)
    //set meta data object type as QR code, here we can add more then one type as well 
    captureMetadataOutput.metadataObjectTypes = [AVMetadataObjectTypeQRCode]


    // Initialize the video preview layer and add it as a sublayer to the viewcontroller view's layer.
    videoPreviewLayer = AVCaptureVideoPreviewLayer(session: captureSession)
    videoPreviewLayer?.videoGravity = AVLayerVideoGravityResizeAspectFill
    videoPreviewLayer?.frame = view.layer.bounds
    view.layer.addSublayer(videoPreviewLayer!)

    // Start capture session.
    captureSession?.startRunning()
} catch {
    // If any error occurs, let the user know. For the example purpose just print out the error
    print(error)
    return
}

3 단계

QR 코드를 읽는 AVCaptureMetadataOutputObjectsDelegate 대리자 메서드 구현

func captureOutput(_ captureOutput: AVCaptureOutput!, didOutputMetadataObjects metadataObjects: [Any]!, from connection: AVCaptureConnection!) {
    
    // Check if the metadataObjects array contains at least one object. If not no QR code is in our video capture
    if metadataObjects == nil || metadataObjects.count == 0 {
        // NO QR code is being detected.
        return
    }
    
    // Get the metadata object and cast it to `AVMetadataMachineReadableCodeObject`
    let metadataObj = metadataObjects[0] as! AVMetadataMachineReadableCodeObject
    
    if metadataObj.type == AVMetadataObjectTypeQRCode {
        // If the found metadata is equal to the QR code metadata then get the string value from meta data
        let barCodeObject = videoPreviewLayer?.transformedMetadataObject(for: metadataObj)
        
        if metadataObj.stringValue != nil {
           // metadataObj.stringValue is our QR code
        }
    }
} 

여기 메타 데이터 객체는 카메라 피드에서 읽은 QR 코드의 범위를 줄 수 있습니다. 경계를 얻으려면 메타 데이터 객체를 아래와 같이 videoPreviewLayertransformedMetadataObject 메소드에 전달하면됩니다.

let barCodeObject = videoPreviewLayer?.transformedMetadataObject(for: metadataObj)
        qrCodeFrameView?.frame = barCodeObject!.bounds


Modified text is an extract of the original Stack Overflow Documentation
아래 라이선스 CC BY-SA 3.0
와 제휴하지 않음 Stack Overflow