美文网首页
Swift(十二):QRScanner 二维码封装总结

Swift(十二):QRScanner 二维码封装总结

作者: IMSong | 来源:发表于2016-08-16 19:25 被阅读378次

封装类 SQRCode.swift

//
//  SQRCode.swift 封装集成类
//  QRCode
//
//  Created by HMC on 16/8/16.
//  Copyright © 2016年 SKing. All rights reserved.
//

import UIKit
import AVFoundation

//闭包的别名
typealias result = (String)->()

class SQRCode: NSObject {
    
    //懒加载 闭包
    lazy var input : AVCaptureInput? = {
        
        //获取摄像头
        let hardWareDevice = AVCaptureDevice.defaultDeviceWithMediaType(AVMediaTypeVideo)
        
        //输入
        guard let input = try? AVCaptureDeviceInput(device: hardWareDevice) else{
            return nil
        }
        return input
        
    }()
    
    lazy var output : AVCaptureMetadataOutput? = {
        //输出
        let output = AVCaptureMetadataOutput()
        //设置输出的代理
        output.setMetadataObjectsDelegate(self, queue: dispatch_get_main_queue())
        
        return output
    }()
    
    lazy var captureLayer : AVCaptureVideoPreviewLayer = {
        
        let captureLayer = AVCaptureVideoPreviewLayer(session: self.session)
        return captureLayer
        
    }()
    
    lazy var session : AVCaptureSession = {
        
        let session = AVCaptureSession()
        return session
    }()
    //结果闭包
    private var resultClosures : result?
    
    //单例
    static let sharedObject = SQRCode()
    
    /**
     实例方法:扫描二维码
     
     - parameter view:   在哪个 VC 的 View
     - parameter result: 闭包输出结果 便于回传到对应的 VC 上
     */
    func scanQRCode(view:UIView , result:(String)->()) {
        
        //
        resultClosures = result
        //连接输入输出
        
        if session.canAddInput(input) && session.canAddOutput(output) {
            session.addInput(input)
            session.addOutput(output)
        }
        
        //输出的二维码识别的码制(所有的码)
        output!.metadataObjectTypes = output!.availableMetadataObjectTypes
        
        
        //设置扫描的区域 0.0-1.0
        //        let screenBounds = UIScreen.mainScreen().bounds
        //        let x :CGFloat = scanBackground.frame.origin.x / screenBounds.size.width
        //        let y :CGFloat = scanBackground.frame.origin.y / screenBounds.size.height
        //        let width :CGFloat = scanBackground.frame.size.width / screenBounds.size.width
        //        let height :CGFloat = scanBackground.frame.size.height / screenBounds.size.height
        //        output.rectOfInterest = CGRectMake(y, x, height, width)
        
        //添加摄像头图层 (其实不是必须的) 防止重复
        let sublayers = view.layer.sublayers!
        guard !sublayers.contains(captureLayer)  else {
            
            return
        }
        
        captureLayer.frame = view.layer.bounds
        view.layer.insertSublayer(captureLayer, atIndex: 0)
        
        // 启动回话: 此时开始采集数据,输出对象
        session.startRunning()
        
        
    }
    
    
    
    /**
     类方法:生成二维码
     
     - parameter string:        字符串
     - parameter iconImageName: 中心的 Icon
     
     - returns: 生成的二维码
     */
    class func createQR(string : String?,iconImageName: String?) -> UIImage? {
        
        //创建二维码滤镜 固定格式 参数必须为 : CIQRCodeGenerator
        let filter = CIFilter(name: "CIQRCodeGenerator")
        
        //复位滤镜,以便下次使用
        filter?.setDefaults()
        
        //KVC 设置输入值  ps:key 固定为inputMessage ,value 必须为 nsdata 类型
        guard string != nil else {
            return nil
        }
        let inputData = string!.dataUsingEncoding(NSUTF8StringEncoding)
        filter?.setValue(inputData, forKey: "inputMessage")
        
        //设置纠错率 一般设置为M
        filter?.setValue("M", forKey: "inputCorrectionLevel")
        
        //接收滤镜返回的图片,由于太小 需要拉伸放大
        guard let image = filter?.outputImage else {
            
            return nil
        }
        //颜色滤镜
        let colorFilter = CIFilter(name: "CIFalseColor")
        colorFilter?.setDefaults()
        colorFilter?.setValue(image, forKey: "inputImage")
        //前景色
        colorFilter?.setValue(CIColor(red: 225/255.0, green: 77/255.0,blue: 0), forKey: "inputColor0")
        //背景色
        colorFilter?.setValue(CIColor(red: 1, green: 1,blue: 1), forKey: "inputColor1")
        
        guard let colorImage = colorFilter?.outputImage else{
            return nil
        }
        //放大25倍 重绘
        let matrix = CGAffineTransformMakeScale(25, 25)
        //将 CIImage 转换为 UIImage
        let QRImage = UIImage(CIImage:colorImage.imageByApplyingTransform(matrix))
        
        //判定 icon 是否为空
        guard iconImageName != nil else {
            
            return nil
        }
        
        //重新绘制中心小图
        if let iconImage = UIImage(named: iconImageName!)  {
            
            let rect = CGRectMake(0, 0, QRImage.size.width, QRImage.size.height)
            UIGraphicsBeginImageContext(rect.size)
            QRImage.drawInRect(rect)
            
            
            let iconSize = CGSizeMake(rect.size.width * 0.25, rect.size.height * 0.25)
            let x = (rect.width - iconSize.width ) * 0.5
            let y = (rect.height - iconSize.height ) * 0.5
            
            iconImage.drawInRect(CGRectMake(x,y,iconSize.width,iconSize.height))
            let resultImage = UIGraphicsGetImageFromCurrentImageContext()
            
            UIGraphicsEndImageContext()
            
            return resultImage
        }
        
        
        return QRImage
    }
    
    
    /**
     类方法:识别二维码
     
     - parameter image: 被识别的图片
     
     - returns: 返回识别的结果
     */
    class func detectorQRFromImage(image:UIImage) -> (String?) {
        //转换为 CIImage
        let ciImage = CIImage(image: image)
        
        //创建探测器
        let detector = CIDetector(ofType: CIDetectorTypeQRCode, context: nil, options: [CIDetectorAccuracy : CIDetectorAccuracyHigh])
        
        //返回结果
        let features = detector.featuresInImage(ciImage!)
        //遍历结果
        for feature in features {
            let qrcode = feature as! CIQRCodeFeature
            
            //print(qrcode.messageString)
            return qrcode.messageString
            
        }
        return nil
        
    }

}


// MARK: - AVCaptureMetadataOutputObjectsDelegate
extension SQRCode : AVCaptureMetadataOutputObjectsDelegate {
    
    func captureOutput(captureOutput: AVCaptureOutput!, didOutputMetadataObjects metadataObjects: [AnyObject]!, fromConnection connection: AVCaptureConnection!) {
        
        guard metadataObjects.count > 0 else{
            
            return
        }
        var resultStr : String?
        
        for obj in metadataObjects {
            guard obj.isKindOfClass(AVMetadataMachineReadableCodeObject) else{
                
                return
            }
            
            let tmpObj = obj as! AVMetadataMachineReadableCodeObject
            
            print(tmpObj.stringValue)
            //print(tmpObj.corners)
            resultStr = tmpObj.stringValue
            
        }
        
        guard (resultClosures != nil) else{
            
            return
        }
        guard resultStr != nil else{
            
            return
        }
        resultClosures!(resultStr!)
    }
}

调用的例子:

例一

//
//  scanQRViewController.swift
//  QRCode
//
//  Created by HMC on 16/8/15.
//  Copyright © 2016年 SKing. All rights reserved.
//

import UIKit
import AVFoundation

class scanQRViewController: UIViewController {
    
    //lazy var session = AVCaptureSession()
    
    /// 扫描网格 距离 框的底部的长度
    @IBOutlet weak var lineToBottom: NSLayoutConstraint!
    /// 扫描框和扫描线的背景 View
    @IBOutlet weak var scanBackground: UIView!
    //扫描框
    @IBOutlet weak var scanDeadLine: UIImageView!
    
    
    override func viewDidLoad() {
        super.viewDidLoad()
        
        // Do any additional setup after loading the view.
        //        startScan()
        
        SQRCode.sharedObject.scanQRCode(view) { (result:String) in
            
            let alert = UIAlertController(title: "识别结果", message: result, preferredStyle: .Alert)
            let action = UIAlertAction(title: "好", style: .Default, handler: { (action: UIAlertAction) in
                self.dismissViewControllerAnimated(true, completion: nil)
            })
            alert.addAction(action)
            
            self.presentViewController(alert, animated: true, completion: nil)
            
        }
    }
    override func viewDidAppear(animated: Bool) {
        
        super.viewDidAppear(animated)
        
        startAnimationOfLine()
        
    }
    
}


// MARK: - 有关的 UI 和动画
extension scanQRViewController{
    /**
     动画的设置
     */
    func startAnimationOfLine() {
        
        //初始化线的位置
        lineToBottom.constant = scanBackground.frame.height
        loadViewIfNeeded()
        
        //设置结束位置
        lineToBottom.constant = -scanBackground.frame.height
        
        //设置动画
        UIView.animateWithDuration(3, animations: {
            //设置重复
            UIView.setAnimationRepeatCount(MAXFLOAT)
            self.view.layoutIfNeeded()
            
        }) { (bool : Bool) in
            
            print("回扫")
            
            
        }
        
    }
    
    
}

例二

//
//  showQRController.swift
//  QRCode
//
//  Created by HMC on 16/8/11.
//  Copyright © 2016年 SKing. All rights reserved.
//

import UIKit
import CoreImage

class showQRController: UIViewController {
    
    @IBOutlet weak var QRImageView: UIImageView!
    
    override func viewDidLoad() {
        super.viewDidLoad()
        
        QRImageView.image = SQRCode.createQR("香雨如酥贵如珠,万家未觉身却无。 一心只为他人事,浓绿艳妆为谁梳?", iconImageName: "123")

        
    }
}

例三

//
//  detectorViewController.swift
//  QRCode
//
//  Created by HMC on 16/8/12.
//  Copyright © 2016年 SKing. All rights reserved.
//

import UIKit

class detectorViewController: UIViewController {
    
    
    @IBOutlet weak var imageView: UIImageView!
    
    override func viewDidLoad() {
        super.viewDidLoad()
        
        // 获得 image
        guard let image = imageView.image else{
            
            return
        }
        
        let returnStr = SQRCode.detectorQRFromImage(image)
        
        //alert 显示识别出的内容
        let alert = UIAlertController(title: "识别内容", message:returnStr, preferredStyle: .Alert)
        let action = UIAlertAction(title: "关闭", style: .Default, handler: { (action : UIAlertAction) in
            self.dismissViewControllerAnimated(true, completion: nil)
        })
        
        alert.addAction(action)
        presentViewController(alert, animated: true, completion: nil)
        
        
        
        
    }
}

相关文章

网友评论

      本文标题:Swift(十二):QRScanner 二维码封装总结

      本文链接:https://www.haomeiwen.com/subject/cmwpsttx.html