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

iOS二维码生成及扫码

现在越来越多的应用加入二维码相关的业务,在iOS开发市场上很多开发人员都在使用第三方的扫码与生成二维码的控件,个人认为此类的第三方控件识别度不高。最近正好整理新框架的事情,研究了一下。具体代码如下

生成二维码代码

/**
 * @author 半  饱, 15-12-18
 *
 * @brief 生成二维码图片
 *
 * @param code  生成二维码图片内容
 * @param width 二维码图片宽度
 * @param height 二维码图片高度
 *
 * @return 返回UIImage对象
 */
- (UIImage *)generateQRCode:(NSString *)code width:(CGFloat)width height:(CGFloat)height {
  CIImage *qrcodeImage;
  NSData *data = [code dataUsingEncoding:NSISOLatin1StringEncoding allowLossyConversion:false];
  CIFilter *filter = [CIFilter filterWithName:@"CIQRCodeGenerator"]; 

  [filter setValue:data forKey:@"inputMessage"];
  [filter setValue:@"H" forKey:@"inputCorrectionLevel"];
  qrcodeImage = [filter outputImage]; 

  CGFloat scaleX = width / qrcodeImage.extent.size.width;
  CGFloat scaleY = height / qrcodeImage.extent.size.height;
  CIImage *transformedImage = [qrcodeImage imageByApplyingTransform:CGAffineTransformScale(CGAffineTransformIdentity, scaleX, scaleY)]; 

  return [UIImage imageWithCIImage:transformedImage];
}

 扫描二维码代码

#import <AVFoundation/AVFoundation.h> 

static const float lightWidth = 240.f;
static const float lightHeight = 240.f;
static const float crossLineWidth = 2.f;
static const float crossLineHeight = 15.f; 

@interface BBScanCodeViewController ()<AVCaptureMetadataOutputObjectsDelegate> {
  float leftWith;
  float topHeight;
}
@property (strong , nonatomic ) AVCaptureDevice *captureDevice;
@property (strong , nonatomic ) AVCaptureDeviceInput *captureInput;
@property (strong , nonatomic ) AVCaptureMetadataOutput *captureOutput;
@property (strong , nonatomic ) AVCaptureSession *captureSession;
@property (strong , nonatomic ) AVCaptureVideoPreviewLayer *capturePreview; 

@property (strong,nonatomic) UIButton *flashLightBtn;
@property (strong,nonatomic) UIImageView *lineImageView; 

@end 

@implementation BBScanCodeViewController
@synthesize captureDevice = _captureDevice;
@synthesize captureInput = _captureInput;
@synthesize captureOutput = _captureOutput;
@synthesize capturePreview = _capturePreview;
@synthesize captureSession = _captureSession;
@synthesize delegate = _delegate;
@synthesize isRectScan = _isRectScan;
@synthesize lineImageView = _lineImageView;
@synthesize flashLightBtn = _flashLightBtn;
- (void)viewDidLoad { 

  [super viewDidLoad];
  self.isShowNavigationItem = YES;
  CGRect screenRect = [UIScreen mainScreen].bounds;
  leftWith = (screenRect.size.width - lightWidth) / 2;
  topHeight =(screenRect.size.height - lightHeight) / 2; 

#if !TARGET_IPHONE_SIMULATOR
  [self initScanCode];
#endif
  [self initLayer];
  [self initViewControl]; 

  [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(willResignActiveNotification) name:UIApplicationWillResignActiveNotification object:nil]; //监听是否触发home键挂起程序.
  [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(didBecomeActiveNotification) name:UIApplicationDidBecomeActiveNotification object:nil]; //监听是否重新进入程序程序.
} 

-(void)viewWillDisappear:(BOOL)animated {
   [self stopScanCode];
  [super viewWillDisappear:animated];
} 

- (void)willResignActiveNotification {
  _flashLightBtn.selected = NO;
}
- (void)didBecomeActiveNotification { 

}
//加载界面上的控件,如:加上闪光灯按钮等
- (void)initViewControl { 

  @autoreleasepool {
    _flashLightBtn = [UIButton buttonWithType:UIButtonTypeCustom];
    [_flashLightBtn setImage:[UIImage imageNamed:@"OpenFlashLight.png"] forState:UIControlStateNormal];
    [_flashLightBtn setImage:[UIImage imageNamed:@"CloseFlashLight.png"] forState:UIControlStateSelected];
    _flashLightBtn.frame = CGRectMake(leftWith, 80.f, 30.f, 30.f);
    [_flashLightBtn addTarget:self action:@selector(systemFlashLight) forControlEvents:UIControlEventTouchUpInside];
    [self.view addSubview:_flashLightBtn]; 

    _lineImageView = [[UIImageView alloc] initWithImage:nil];
    _lineImageView.backgroundColor = [UIColor greenColor];
    _lineImageView.frame = CGRectMake(leftWith, topHeight, lightWidth, 2);
    [self.view addSubview:_lineImageView];
    [self scanLineAnimation];
  } 

} 

- (void)scanLineAnimation {
  [UIView beginAnimations:nil context:nil];
  [UIView setAnimationDuration:4.f];
  //设置代理
  [UIView setAnimationDelegate:self];
  //设置动画执行完毕调用的事件
  [UIView setAnimationDidStopSelector:@selector(didViewAnimation)];
  _lineImageView.frame = CGRectMake(leftWith,topHeight + lightHeight-2,lightWidth,2);
  [UIView commitAnimations]; 

} 

-(void)didViewAnimation {
//  self.navigationController
  _lineImageView.frame = CGRectMake(leftWith, topHeight, lightWidth, 2);
  [self scanLineAnimation];
} 

- (void)insertLayerWithFrame:(CGRect)frame withBackgroundColor:(UIColor *)backgroundColor {
  @autoreleasepool {
    CALayer *layer = [CALayer layer];
    layer.backgroundColor = backgroundColor.CGColor;
    layer.frame = frame;
    [self.view.layer addSublayer:layer];
  }
}
//初始化layer层,绘制半透明区域
-(void) initLayer {
  //公共参数
  UIColor *fillColor = [UIColor colorWithRed:0xae/255.f green:0xae/255.f blue:0xae/255.f alpha:0.4];
  UIColor *crossColor = [UIColor greenColor];
  CGRect screenRect = [UIScreen mainScreen].bounds;
  [self insertLayerWithFrame:CGRectMake(0, 0, leftWith, screenRect.size.height) withBackgroundColor:fillColor];
  [self insertLayerWithFrame:CGRectMake(leftWith, 0, lightWidth, topHeight) withBackgroundColor:fillColor];
  [self insertLayerWithFrame:CGRectMake(leftWith + lightWidth, 0, leftWith, screenRect.size.height) withBackgroundColor:fillColor];
  [self insertLayerWithFrame:CGRectMake(leftWith, topHeight + lightHeight, lightWidth, topHeight) withBackgroundColor:fillColor]; 

  [self insertLayerWithFrame:CGRectMake(leftWith, topHeight, crossLineWidth, crossLineHeight) withBackgroundColor:crossColor];
  [self insertLayerWithFrame:CGRectMake(leftWith, topHeight, crossLineHeight, crossLineWidth) withBackgroundColor:crossColor]; 

  [self insertLayerWithFrame:CGRectMake(leftWith + lightWidth - crossLineHeight, topHeight, crossLineHeight, crossLineWidth) withBackgroundColor:crossColor];
  [self insertLayerWithFrame:CGRectMake(leftWith + lightWidth - crossLineWidth, topHeight, crossLineWidth, crossLineHeight) withBackgroundColor:crossColor]; 

  [self insertLayerWithFrame:CGRectMake(leftWith, topHeight + lightHeight - crossLineHeight, crossLineWidth, crossLineHeight) withBackgroundColor:crossColor];
  [self insertLayerWithFrame:CGRectMake(leftWith, topHeight + lightHeight - crossLineWidth, crossLineHeight, crossLineWidth) withBackgroundColor:crossColor]; 

  [self insertLayerWithFrame:CGRectMake(leftWith + lightWidth - crossLineHeight, topHeight + lightHeight - crossLineWidth, crossLineHeight, crossLineWidth) withBackgroundColor:crossColor];
  [self insertLayerWithFrame:CGRectMake(leftWith + lightWidth - crossLineWidth, topHeight + lightHeight - crossLineHeight, crossLineWidth, crossLineHeight) withBackgroundColor:crossColor];
} 

-(void)initScanCode {
  @autoreleasepool {
    _captureDevice = [ AVCaptureDevice defaultDeviceWithMediaType : AVMediaTypeVideo];
    _captureInput = [ AVCaptureDeviceInput deviceInputWithDevice : _captureDevice error : nil ];
    _captureOutput = [[ AVCaptureMetadataOutput alloc ] init ];
    [_captureOutput setMetadataObjectsDelegate : self queue : dispatch_get_main_queue ()];
    if (_isRectScan) {
      CGRect screenRect = [UIScreen mainScreen].bounds;
      [ _captureOutput setRectOfInterest : CGRectMake (topHeight / screenRect.size.height, leftWith / screenRect.size.width, lightHeight/screenRect.size.height, lightWidth / screenRect.size.width)];
    } 

    _captureSession = [[ AVCaptureSession alloc ] init ];
    [_captureSession setSessionPreset : AVCaptureSessionPresetHigh ];
    if ([_captureSession canAddInput : _captureInput ])
    {
      [_captureSession addInput : _captureInput ];
    }
    if ([_captureSession canAddOutput : _captureOutput ])
    {
      [_captureSession addOutput : _captureOutput ];
    }
    _captureOutput.metadataObjectTypes = @[AVMetadataObjectTypeQRCode ] ; 

    _capturePreview =[ AVCaptureVideoPreviewLayer layerWithSession :_captureSession ];
    _capturePreview.videoGravity = AVLayerVideoGravityResizeAspectFill ;
    _capturePreview.frame = self.view.layer.bounds ;
    [self.view.layer insertSublayer : _capturePreview atIndex : 0 ];
    [_captureSession startRunning ];
  }
} 

- ( void )captureOutput:( AVCaptureOutput *)captureOutput didOutputMetadataObjects:( NSArray *)metadataObjects fromConnection:( AVCaptureConnection *)connection
{
  if (metadataObjects != nil && [metadataObjects count] > 0) {
    AVMetadataMachineReadableCodeObject *metadataObj = [metadataObjects objectAtIndex:0];
    NSString *scanCodeResult;
    if ([[metadataObj type] isEqualToString:AVMetadataObjectTypeQRCode]) {
      [self stopScanCode];
      scanCodeResult = metadataObj.stringValue;
      //回调信息
      if (_delegate && [_delegate respondsToSelector:@selector(scanCodeResultByViewController:withScanCodeResult:)]) {
        [_delegate scanCodeResultByViewController:self withScanCodeResult:scanCodeResult];
        [self.navigationController popViewControllerAnimated:YES];
      }
    } else {
      NSLog(@"扫描信息错误!");
    }
  }
} 

- (void)systemFlashLight
{
#if !TARGET_IPHONE_SIMULATOR
  if([_captureDevice hasTorch] && [self.captureDevice hasFlash])
  {
    [_captureSession beginConfiguration];
    [_captureDevice lockForConfiguration:nil];
    if(_captureDevice.torchMode == AVCaptureTorchModeOff)
    {
      _flashLightBtn.selected = YES;
      [_captureDevice setTorchMode:AVCaptureTorchModeOn];
      [_captureDevice setFlashMode:AVCaptureFlashModeOn];
    }
    else {
      _flashLightBtn.selected = NO;
      [_captureDevice setTorchMode:AVCaptureTorchModeOff];
      [_captureDevice setFlashMode:AVCaptureFlashModeOff];
    }
    [_captureDevice unlockForConfiguration];
    [_captureSession commitConfiguration];
  }
#else
  [CommonUtil showAlert:G_ALERTTITLE withMessage:@"虚拟设备不能运行摄像头!"];
#endif
} 

-(void)stopScanCode {
  [_captureSession stopRunning];
  _captureSession = nil;
  _captureDevice = nil;
  _captureInput = nil;
  _captureOutput = nil;
  [_capturePreview removeFromSuperlayer];
} 

- (void)didReceiveMemoryWarning {
  [super didReceiveMemoryWarning];
} 

@end

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

(0)

相关推荐

  • 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自带原生二维码扫描的实现

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

  • ios原生二维码扫描

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

  • iOS二维码的生成代码

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

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

    基于swift3.0 1.扫描二维码 设置扫描会话,图层和输入输出 //设置捕捉设备 let device = AVCaptureDevice.defaultDevice(withMediaType: AVMediaTypeVideo) do { //设置设备输入输出 let input = try AVCaptureDeviceInput(device: device) let output = AVCaptureMetadataOutput() output.setMetadataObjec

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

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

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

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

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

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

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

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

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

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

随机推荐