ios swift3.0实现二维码扫描、生成、识别示例代码

基于swift3.0

1.扫描二维码

设置扫描会话,图层和输入输出

  //设置捕捉设备
    let device = AVCaptureDevice.defaultDevice(withMediaType: AVMediaTypeVideo)

    do
    {
      //设置设备输入输出
      let input = try AVCaptureDeviceInput(device: device)

      let output = AVCaptureMetadataOutput()
      output.setMetadataObjectsDelegate(self, queue: DispatchQueue.main)

      //设置会话
      let scanSession = AVCaptureSession()
      scanSession.canSetSessionPreset(AVCaptureSessionPresetHigh)

      if scanSession.canAddInput(input)
      {
        scanSession.addInput(input)
      }

      if scanSession.canAddOutput(output)
      {
        scanSession.addOutput(output)
      }

      //设置扫描类型(二维码和条形码)
      output.metadataObjectTypes = [
      AVMetadataObjectTypeQRCode,
      AVMetadataObjectTypeCode39Code,
      AVMetadataObjectTypeCode128Code,
      AVMetadataObjectTypeCode39Mod43Code,
      AVMetadataObjectTypeEAN13Code,
      AVMetadataObjectTypeEAN8Code,
      AVMetadataObjectTypeCode93Code]

      //预览图层
      let scanPreviewLayer = AVCaptureVideoPreviewLayer(session:scanSession)
      scanPreviewLayer?.videoGravity = AVLayerVideoGravityResizeAspectFill
      scanPreviewLayer?.frame = view.layer.bounds

      view.layer.insertSublayer(scanPreviewLayer!, at: 0)

      //自动对焦
      if (device?.isFocusModeSupported(.autoFocus))!
      {
        do { try input.device.lockForConfiguration() } catch{ }
        input.device.focusMode = .autoFocus
        input.device.unlockForConfiguration()
      }

      //设置扫描区域
      NotificationCenter.default.addObserver(forName: NSNotification.Name.AVCaptureInputPortFormatDescriptionDidChange, object: nil, queue: nil, using: {[weak self] (noti) in
          output.rectOfInterest = (scanPreviewLayer?.metadataOutputRectOfInterest(for: self!.scanPane.frame))!
      })

      //保存会话
      self.scanSession = scanSession

    }
    catch
    {
      //摄像头不可用

      Tool.confirm(title: "温馨提示", message: "摄像头不可用", controller: self)

      return
    }

开始扫描

    if !scanSession.isRunning
    {
      scanSession.startRunning()
    }

扫描结果在代理方法中

//扫描捕捉完成
extension ScanCodeViewController : AVCaptureMetadataOutputObjectsDelegate
{

  func captureOutput(_ captureOutput: AVCaptureOutput!, didOutputMetadataObjects metadataObjects: [Any]!, from connection: AVCaptureConnection!)
  {

    //停止扫描
    self.scanLine.layer.removeAllAnimations()
    self.scanSession!.stopRunning()

    //播放声音
    Tool.playAlertSound(sound: "noticeMusic.caf")

    //扫完完成
    if metadataObjects.count > 0
    {

      if let resultObj = metadataObjects.first as? AVMetadataMachineReadableCodeObject
      {

        Tool.confirm(title: "扫描结果", message: resultObj.stringValue, controller: self,handler: { (_) in
          //继续扫描
          self.startScan()
        })

      }

    }

  }

}

2.二维码生成

通过滤镜生成CGImage

    //2.二维码滤镜
    let contentData = self.data(using: String.Encoding.utf8)
    let fileter = CIFilter(name: "CIQRCodeGenerator")

    fileter?.setValue(contentData, forKey: "inputMessage")
    fileter?.setValue("H", forKey: "inputCorrectionLevel")

    let ciImage = fileter?.outputImage

    //3.颜色滤镜
    let colorFilter = CIFilter(name: "CIFalseColor")

    colorFilter?.setValue(ciImage, forKey: "inputImage")
    colorFilter?.setValue(CIColor(cgColor: QRCodeColor.cgColor), forKey: "inputColor0")// 二维码颜色
    colorFilter?.setValue(CIColor(cgColor: QRCodeBgColor.cgColor), forKey: "inputColor1")// 背景色

    //4.生成处理

    let outImage = colorFilter!.outputImage
    let scale = QRCodeSize / outImage!.extent.size.width;

    let transform = CGAffineTransform(scaleX: scale, y: scale)

    let transformImage = colorFilter!.outputImage!.applying(transform)

通过CGImage生成UIImage

let image = UIImage(ciImage: ciImage)

绘制Logo和边框

// 绘制logo
UIGraphicsBeginImageContextWithOptions(image.size, false, UIScreen.main.scale)
    image.draw(in: CGRect(x: 0, y: 0, width: image.size.width, height: image.size.height))

//线框
let logoBorderLineImagae = QRCodeLogo.getRoundRectImage(size: logoWidth, radius: radius, borderWidth: borderLineWidth, borderColor: borderLineColor)
//边框
let logoBorderImagae = logoBorderLineImagae.getRoundRectImage(size: logoWidth, radius: radius, borderWidth: boderWidth, borderColor: borderColor)

logoBorderImagae.draw(in: logoFrame)

let QRCodeImage = UIGraphicsGetImageFromCurrentImageContext()
UIGraphicsEndImageContext()

封装接口:

 /**
   1.生成二维码

   - returns: 黑白普通二维码(大小为300)
   */

  func generateQRCode() -> UIImage

    /**
   2.生成二维码

   - parameter size: 大小

   - returns: 生成带大小参数的黑白普通二维码
   */
   func generateQRCodeWithSize(size:CGFloat?) -> UIImage

     /**
   3.生成二维码

   - parameter logo: 图标

   - returns: 生成带Logo二维码(大小:300)
   */
   func generateQRCodeWithLogo(logo:UIImage?) -> UIImage

     /**
   4.生成二维码

   - parameter size: 大小
   - parameter logo: 图标

   - returns: 生成大小和Logo的二维码
   */
  func generateQRCode(size:CGFloat?,logo:UIImage?) -> UIImage

    /**
   5.生成二维码

   - parameter size:  大小
   - parameter color:  颜色
   - parameter bgColor: 背景颜色
   - parameter logo:  图标

   - returns: 带Logo、颜色二维码
   */
  func generateQRCode(size:CGFloat?,color:UIColor?,bgColor:UIColor?,logo:UIImage?) -> UIImage

    /**
   6.生成二维码

   - parameter size:      大小
   - parameter color:      颜色
   - parameter bgColor:     背景颜色
   - parameter logo:      图标
   - parameter radius:     圆角
   - parameter borderLineWidth: 线宽
   - parameter borderLineColor: 线颜色
   - parameter boderWidth:   带宽
   - parameter borderColor:   带颜色

   - returns: 自定义二维码
   */
  func generateQRCode(size:CGFloat?,color:UIColor?,bgColor:UIColor?,logo:UIImage?,radius:CGFloat,borderLineWidth:CGFloat?,borderLineColor:UIColor?,boderWidth:CGFloat?,borderColor:UIColor?) -> UIImage
使用

DispatchQueue.global().async {

let image = content.generateQRCodeWithLogo(logo: self.logoImageView.image)
        DispatchQueue.main.async(execute: {
          self.QRCodeImageView.image = image
        })

      }

3.识别二维码

通过CIDetector识别二维码

CIDetector用于分析CIImage,以得到CIFeature,每个CIDetector都要用一个探测器类型(NSString)来初始化。这个类型用于告诉探测器要找什么特征

1.识别图片二维码

  func recognizeQRCode() -> String?
  {

    let detector = CIDetector(ofType: CIDetectorTypeQRCode, context: nil, options: [CIDetectorAccuracy : CIDetectorAccuracyHigh])
    let features = detector?.features(in: CoreImage.CIImage(cgImage: self.cgImage!))
    guard (features?.count)! > 0 else { return nil }
    let feature = features?.first as? CIQRCodeFeature
    return feature?.messageString

  }

使用实例

DispatchQueue.global().async {
      let recognizeResult = self.sourceImage?.recognizeQRCode()
      let result = recognizeResult?.characters.count > 0 ? recognizeResult : "无法识别"
      DispatchQueue.main.async {
        Tool.confirm(title: "扫描结果", message: result, controller: self)
        self.activityIndicatoryView.stopAnimating()
      }
    }

本文Demo地址:QRCode_jb51.rar

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • iOS 二维码生成及扫码详解及实例代码

    iOS二维码生成及扫码 现在越来越多的应用加入二维码相关的业务,在iOS开发市场上很多开发人员都在使用第三方的扫码与生成二维码的控件,个人认为此类的第三方控件识别度不高.最近正好整理新框架的事情,研究了一下.具体代码如下 生成二维码代码 /** * @author 半 饱, 15-12-18 * * @brief 生成二维码图片 * * @param code 生成二维码图片内容 * @param width 二维码图片宽度 * @param height 二维码图片高度 * * @return

  • iOS实现二维码的扫描功能

    直接上代码,就不多废话了 // // ViewController.m // QRCode // // Created by chenchen on 15/7/30. // Copyright (c) 2015年 BSY. All rights reserved. // #import <AVFoundation/AVFoundation.h> #import "ViewController.h" @interface ViewController ()<AVCapt

  • ios原生二维码扫描

    做iOS的二维码扫描,有两个第三方库可以选择,ZBar和ZXing.今天要介绍的是iOS7.0后AVFoundation框架提供的原生二维码扫描. 首先需要添加AVFoundation.framework框架到你工程中build phase的"Link Binary With Libraries"之下,然后就可以开始了. 一.做好准备工作,搭建UI UI效果如图 IBOutlet.IBAction如下: @property (weak, nonatomic) IBOutlet UIVi

  • 如何在iphon IOS设备上使用二维码

    下面给大家介绍下二维码简介 二维码 (2-dimensional bar code) 是用某种特定的几何图形按一定规律在平面(二维方向上)分布的黑白相间的图形记录数据符号信息的:在代码编制上巧妙地利用构成计算机内部逻辑基础的"0"."1"比特流的概念,使用若干个与二进制相对应的几何形体来表示文字数值信息,通过图象输入设备或光电扫描设备自动识读以实现信息自动处理: 二维条码/二维码能够在横向和纵向两个方位同时表达信息,因此能在很小的面积内表达大量的信息. 下面介绍下如

  • IOS笔记061之二维码的生成和扫描

    如今二维码随处可见,无论是实物商品还是各种礼券都少不了二维码的身影.而手机等移动设备又成为二维码的一个很好的应用平台,不管是生成二维码还是扫码二维码.本篇文章从生成二维码.扫描二维码展开分析,通过内容分析二维码用起来也很easy了. 首先说下生成二维码 二维码可以存放纯文本.名片或者URL 其次生成二维码的步骤: 导入CoreImage框架 再次通过滤镜CIFilter生成二维码 1.创建过滤器 2.恢复滤镜的默认属性 3.设置内容 4.获取输出文件 5.显示二维码 代码实现 CoreImage

  • iOS自带原生二维码扫描的实现

    前言 首先说明的是:原生的二维码扫描有一个坑,那就是扫描范围的确定.只要记得扫描范围是X与Y互换位置,W与H互换位置,就没有什么问题了. 下面进入正题: 1.因为使用原生二维码扫描,所以需要加入头文件添加delegate #import <AVFoundation/AVFoundation.h> <AVCaptureMetadataOutputObjectsDelegate> 2.接着是使用到的类 @property (strong,nonatomic)AVCaptureDevic

  • iOS二维码的生成代码

    本文实例为大家分享了iOS二维码的生成代码,供大家参考,具体内容如下 一.工程图. 二.代码. ViewController.m #import "ViewController.h" #import "ScanViewController.h" @interface ViewController () @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; // D

  • iOS 条码及二维码扫描(从相册中读取条形码/二维码)及扫码过程中遇到的坑

    文章重点介绍如何解决,从手机相册中读取条形码和二维码的问题 1.扫码. 网上有特别的关于iOS扫码的代码和示例,其中扫码主要使用的是自带的AVFoundation类.这里就不细说了,要注意的是如何设置扫描区域,识别区域(这个值是按比例0~1设置,而且X.Y要调换位置,width.height调换位置) <span style="font-size:14px;">//创建输出流 AVCaptureMetadataOutput * output = [[AVCaptureMet

  • iOS和Android用同一个二维码实现跳转下载链接的方法

    前言 最近一个项目需要iOS和安卓使用一个二维码,让扫描的机器自己识别操作系统实现跳转到相应的下载链接.比如iPhone用微信进行扫描就让他跳转appStore的下载页面,安卓机器使用微信扫描就直接跳浏览器下载.但是这二维码还有一个需求就是,用户已经下载了这个app,当用户打开app进入到注册页面时,再次扫描这个二维码时,自动填写邀请码进行注册.那么该如何实现,细节就不说了,直接上代码. 使用js实现,其实代码非常简单. 使用时直接拷贝代码,改掉相应的链接就好. PS:该链接在微信环境打开时还是

  • IOS生成与读取二维码名片

    一.概述 1.通过某种特定的几何图形按一定规律在平面(二维方向上)分布的黑白相间的图形记录数据符号信息的 2.二维码通常可以包含以下内容 纯文本 名片 URL 3.二维码具有非常广泛的应用 二维码名片 扫码付款 网址(URL),扫描后自动打开网址 二.二维码的生成 1.生成原理 通过一个类CIFilter(滤镜)包含二维码中所有的信息,然后生成一张二维码图片 二维码中间的icon(头像),是通过在生成的图片上添加一张图片实现的 2.生成步骤 创建滤镜 初始化滤镜 添加二维码信息 获取生成的二维码

随机推荐