ios-深度解析二维码的生成与使用

前言:利用一个小demo来对二维码进行学习,总共四个界面(主界面,生成二维码界面,识别二维码界面,扫描二维码界面)

想要源码点这里:QR-code

一.二维码的介绍

1.什么是二维码?

二维条码/二维码是用某种特定的几何图形按一定规律在平面分布的黑白相间的图形记录数据符号信息的

总结: 用图形记录标记一些信息,方便通过图形识别来获取信息

2 应用场景

信息获取(名片、地图、WIFI密码、资料)
手机电商(用户扫码、手机直接购物下单)
手机支付(扫描商品二维码,通过银行或第三方支付提供的手机端通道完成支付)
微信添加好友

二.二维码界面的搭建

1.总共四个界面,可以采用storyBoard来搭建

2.四个storyBoard放在一个界面,看起来不美观,还很容易搞混,有没有优化方案?
可以把四个storyBoard分别开来,单独放到一个界面里面

3.怎么把storyBoard单独放在一个界面,而且还让这些界面有联系(连线)?

可以用storyBoard reference 来解决 就是用一个引用来代替storyBoard,保持storyBoard间的联系(连线)

4.最终效果


三.二维码的生成

1.生成二维码的步骤

1.1 创建滤镜 CIFilter
滤镜属于CoreImage框架,要导入该框架 该框架将常用来处理图片(生成毛玻璃效果/二维码)

1.2 给滤镜设置内容(用kvc方式赋值)
内容必须为NSData类型

1.3 获取生成的二维码图片
获取的图片是CIImage类型的,使用的话要进行转换


2.运行程序发现生成的二维码图片很模糊,为什么?

生成为二维码图片大小为 27 * 27 被拉伸的太大,所以不清晰

3.怎么显示清晰的二维码?

苹果提供一个api对图片放大,还不影响清晰度

  // 1.创建Transform
    let scale = imageView.bounds.width / orginalImage.extent.width
    let transform = CGAffineTransformMakeScale(scale, scale)

    // 2.放大图片
    let hdImage = orginalImage.imageByApplyingTransform(transform)

4.设置前景图片

4.1 为什么要设置前景图片?
一般二维码中心都有一张小的图片,就是前景图片
生成二维码没有前景图片,需要手动添加前景图片

4.2 怎么添加前景图片?
就是把两张图片合成为一张图片,用绘图就可以轻松搞定

4.3 绘图的步骤
4.31 开启图形上下文
4.32 将二维码图片画到图形上下文(二维码的size = 图形上下文的size)
4.33 将前景图片画到图形上下文(前景图片的center = 图形上下文的center)
4.34 从图形上下文获取新的图片
4.35 关闭图形上下文

四.二维码的识别

1.获取相册中的二维码

1.1 怎么获取相册?
1.11 创建照片选择控制器
1.12 设置照片的来源类型
1.13 设置代理
1.14 弹出控制器

     // 1.创建照片选择控制器
    let ipc = UIImagePickerController()
    // 2.设置来源的类型
    ipc.sourceType = .PhotoLibrary
    // 3.设置代理
    ipc.delegate = self
    // 4.弹出控制器
    presentViewController(ipc, animated: true, completion: nil)

      在代理方法中实现    选中图片dismiss掉控制器
func imagePickerController(picker: UIImagePickerController,   
didFinishPickingMediaWithInfo info: [String : AnyObject]) {

      //选中照片
    imageView.image = info[UIImagePickerControllerOriginalImage] as? UIImage

    picker.dismissViewControllerAnimated(true, completion: nil)
}

2.识别二维码的步骤

2.1 创建 CIDetector对象(识别器)
2.2 获取图片,并将图片转成 CIIImage
2.3 识别图片中的二维码(得到一个数组,图片中可能有多个二维码)
2.4 遍历数组

   // 1.创建识别器
    let detector = CIDetector(ofType: CIDetectorTypeQRCode, context: nil, options: nil)

    // 2.获取图片,并且将图片转成CIIImage
    let image = imageView.image!
    guard let ciImage = CIImage(image: image) else {
        return
    }
    // 3.识别图片中二维码
    let features = detector.featuresInImage(ciImage)

    // 4.遍历数组中所有的元素
    for f in features {
        guard let qrCodeF = f as? CIQRCodeFeature else {
            continue
        }
        //打印二维码的信息
        print(qrCodeF.messageString)
    }

五.扫描二维码(需要真机操作)

1.扫描二维码界面搭建

1.1 主要就是扫描框的搭建
扫描框中再加上一个ImageView,给ImageView一个动画 模拟正在扫描(冲击波)

1.2 扫描框是一张图片,冲击波也是一张图片,他们的位置和尺寸是一样的
为了方便以后更改控件的位置,可以用一个view把扫描框和冲击波封装在里面

1.3 扫描动画(冲击波)动画怎么做?
1.31 设置冲击波的底部约束相对于父控件(view)有一个间距
1.32 更改约束的间距,来达到动画的效果

    // 1.改变约束(原来约束为-240)
    scanViewBottomCons.constant = 240
    // 2.执行动画
    UIView.animateWithDuration(1.0) {
        UIView.setAnimationRepeatCount(MAXFLOAT)
        self.qrCodeView.layoutIfNeeded()
    }

2.扫描二维码

2.1 扫描步骤
2.11 创建捕捉会话(需要导入AVFoundation框架)
2.12 设置输入(摄像头)
2.13 设置输出 Metadata
2.14 添加预览图层(可以没有)
预览图层是为了让用户知道扫描到哪里了,一般为了用户体验,都会添加
2.15 开始扫描

源代码:建议不要死记,用的时候直接拷贝
     // 1.创建捕捉会话
    let session = AVCaptureSession()

    // 2.设置输入(摄像头)
    let device = AVCaptureDevice.defaultDeviceWithMediaType(AVMediaTypeVideo)
    guard let input = try? AVCaptureDeviceInput(device: device) else {
        return
    }
    session.addInput(input)

    // 3.设置输出(Metadata)
    let output = AVCaptureMetadataOutput()
    // 设置代理
    output.setMetadataObjectsDelegate(self, queue: dispatch_get_main_queue())
    session.addOutput(output)
    // 设置output的输出的类型(该类型的设置必须在添加到session之后)
    output.metadataObjectTypes = [AVMetadataObjectTypeQRCode]

    // 4.添加预览图层(可以没有)
    let previewLayer = AVCaptureVideoPreviewLayer(session: session)
    previewLayer.frame = view.bounds
    view.layer.insertSublayer(previewLayer, atIndex: 0)

    // 5.开始扫描
    session.startRunning()

3.获取扫描结果

3.1设置代理,在代理方法中拿到结果
代理方法

func captureOutput(captureOutput: AVCaptureOutput!, 
didOutputMetadataObjects metadataObjects: [AnyObject]!, 
fromConnection connection: AVCaptureConnection!) {

  guard let objc = metadataObjects.last as? AVMetadataMachineReadableCodeObject 

        else {
                  return
              }
              print(objc.stringValue)
          }

5.设置扫描区域.

5.1 为什么要设置扫描区域?

扫描二维码,发现只要二维码进入摄像头区域,就能直接扫描

要求是进入扫描框,才进行扫描

5.2 怎么设置扫描区域?

通过设置 output.rectOfInterest属性来设置扫描区域

    // 设置扫描的区域
    let screenW = UIScreen.mainScreen().bounds.width
    let screenH = UIScreen.mainScreen().bounds.height
    let x : CGFloat = qrCodeView.frame.origin.x / screenW
    let y : CGFloat = qrCodeView.frame.origin.y / screenH
    let w : CGFloat = qrCodeView.frame.width / screenW
    let h : CGFloat = qrCodeView.frame.height / screenH
    output.rectOfInterest = CGRect(x: y, y: x, width: h, height: w)

注意:扫描区域的坐标系与屏幕的坐标系正好相反 ( 扫描区域x = 屏幕坐标系 y)

5.3 设置扫描区域代码写到哪里?

扫描区域属于 输出的一个属性,应该写到创建输出代码的后边

想要源码点这里:QR-code

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,921评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,635评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,393评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,836评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,833评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,685评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,043评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,694评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,671评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,670评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,779评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,424评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,027评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,984评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,214评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,108评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,517评论 2 343

推荐阅读更多精彩内容

  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,019评论 4 62
  • 在一起的时间长了,似乎故事就可以一直这样写下去,细细碎碎。许多甜蜜幸福的片段都值得铭记。又或者,只要是一起经历的时...
    mo清夜无尘阅读 282评论 1 2
  • 你有没有过那么一瞬间,感觉无论四周环绕着多少嬉笑怒骂的人,无论有多么亲密无间的朋友陪伴在你身边,你依然觉得孤独。 ...
    冬柯一梦_BTS_army阅读 392评论 0 0
  • 被物欲左右的人往往只有小聪明,层次较低。他们没办法拒绝享乐与奢靡,没办法抗拒所有肤浅的事物,所以他们不能静心静气的...
    享受生活的Catherine阅读 160评论 0 0
  • 揖别乡关岁月深, 少年脚印可留痕? 如今白发飘飘矣, 一梦依然在老村。
    高湛明阅读 514评论 5 5