iOS开发-原生二维码的扫描和生成

从iOS7开始官方集成了二维码的扫描和生成功能
此前被广泛使用的zbarsdk目前不支持64位处理器

1.扫描二维码
扫描二维码需要导入AVFoundation框架
利用摄像头识别二维码中的内容(模拟器不行)

输入(摄像头)
由会话将摄像头采集到的二维码图像转换成字符串数据
输出(数据)
由预览图层显示扫描场景

// 实例化拍摄设备

AVCaptureDevice *device = [AVCaptureDevice defaultDeviceWithMediaType:AVMediaTypeVideo];

// 设置输入设备

AVCaptureDeviceInput *input = [AVCaptureDeviceInput deviceInputWithDevice:device error:nil];

// 设置元数据输出

// 实例化拍摄元数据输出

AVCaptureMetadataOutput *output = [[AVCaptureMetadataOutput alloc] init];

// 设置输出数据代理

[output setMetadataObjectsDelegate:self queue:dispatch_get_main_queue()];

// 添加拍摄会话

// 实例化拍摄会话

AVCapturesession *session = [[AVCaptureSession alloc] init];

// 添加会话输入

[session addInput:input];

// 添加会话输出

[session addOutput:output];

// 设置输出数据类型,需要将元数据输出添加到会话后,才能指定元数据类型,否则会报错

[output setMetadataObjectTypes:@[AVMetadataObjectTypeQRCode]];

// 视频预览图层

// 实例化预览图层

AVCaptureVideoPReviewLayer *preview = [AVCaptureVideoPreviewLayer layerWithSession:_session];

preview.videoGravity = AVLayerVideoGravityResizeaspectFill;

preview.frame = self.view.bounds;

// 将图层插入当前视图

[self.view.layer insertSublayer:preview atIndex:100];

self.previewLayer = preview;

// 启动会话

[_session startRunning];

2.扫描二维码实例
@interface ViewController ()

@property (nonatomic, strong) AVCaptureSession *session;

@property (nonatomic, strong) AVCaptureVideoPreviewLayer *previewLayer;

@end

@implementation ViewController

  • (void)viewDidLoad

{

[super viewDidLoad];

// 1. 实例化拍摄设备

AVCaptureDevice *device = [AVCaptureDevice defaultDeviceWithMediaType:AVMediaTypeVideo];

// 2. 设置输入设备

AVCaptureDeviceInput *input = [AVCaptureDeviceInput deviceInputWithDevice:device error:nil];

// 3. 设置元数据输出

// 3.1 实例化拍摄元数据输出

AVCaptureMetadataOutput *output = [[AVCaptureMetadataOutput alloc] init];

// 3.3 设置输出数据代理

[output setMetadataObjectsDelegate:self queue:dispatch_get_main_queue()];

// 4. 添加拍摄会话

// 4.1 实例化拍摄会话

AVCaptureSession *session = [[AVCaptureSession alloc] init];

// 4.2 添加会话输入

[session addInput:input];

// 4.3 添加会话输出

[session addOutput:output];

// 4.3 设置输出数据类型,需要将元数据输出添加到会话后,才能指定元数据类型,否则会报错

[output setMetadataObjectTypes:@[AVMetadataObjectTypeQRCode]];

self.session = session;

// 5. 视频预览图层

// 5.1 实例化预览图层, 传递_session是为了告诉图层将来显示什么内容

AVCaptureVideoPreviewLayer *preview = [AVCaptureVideoPreviewLayer layerWithSession:_session];

preview.videoGravity = AVLayerVideoGravityResizeAspectFill;

preview.frame = self.view.bounds;

// 5.2 将图层插入当前视图

[self.view.layer insertSublayer:preview atIndex:100];

self.previewLayer = preview;

// 6. 启动会话

[_session startRunning];

}

  • (void)captureOutput:(AVCaptureOutput )captureOutput didOutputMetadataObjects:(NSArray )metadataObjects fromConnection:(AVCaptureConnection *)connection {

// 会频繁的扫描,调用代理方法

// 1. 如果扫描完成,停止会话

[self.session stopRunning];

// 2. 删除预览图层

[self.previewLayer removeFromSuperlayer];

NSLog(@”%@”, metadataObjects);

// 3. 设置界面显示扫描结果

if (metadataObjects.count > 0)

{

AVMetadataMachineReadableCodeObject *obj = metadataObjects[0];

// 提示:如果需要对url或者名片等信息进行扫描,可以在此进行扩展!

NSLog(@”%@”, obj.stringValue);

}

}

@end

3.二维码的生成
生成二维码的步骤:
导入CoreImage框架
通过滤镜CIFilter生成二维码

二维码的内容(传统的条形码只能放数字):
纯文本,
名片,
URL

// 1. 实例化二维码滤镜

CIFilter *filter = [CIFilter filterWithName:@”CIQRCodeGenerator”];

// 2. 恢复滤镜的默认属性

[filter setDefaults];

// 3. 将字符串转换成

NSData NSData *data = [str dataUsingEncoding:NSUTF8StringEncoding];

// 4. 通过KVO设置滤镜inputMessage数据

[filter setValue:data forKey:@”inputMessage”];

// 5. 获得滤镜输出的图像

CIImage *outputImage = [filter outputImage];

// 6. 将CIImage转换成UIImage,并放大显示

return [UIImage imageWithCIImage:outputImage scale:20.0 orientation:UIImageOrientationUp];

4.生成二维码实例
- (void)viewDidLoad {

[super viewDidLoad];

// Do any additional setup after loading the view, typically from a nib.



self.imgView=[[UIImageView alloc]initWithFrame:CGRectMake(100, 100, [UIScreen mainScreen].bounds.size.width/2.0, [UIScreen mainScreen].bounds.size.width/2.0)];

[self.view addSubview:_imgView];

[self erweima];

}

-(void)erweima

{

//二维码滤镜

CIFilter *filter=[CIFilter filterWithName:@"CIQRCodeGenerator"];

//恢复滤镜的默认属性

[filter setDefaults];

//将字符串转换成NSData

NSData *data=[@"www.baidu.com" dataUsingEncoding:NSUTF8StringEncoding];

//通过KVO设置滤镜inputmessage数据

[filter setValue:data forKey:@"inputMessage"];

//获得滤镜输出的图像

CIImage *outputImage=[filter outputImage];

//将CIImage转换成UIImage,并放大显示

_imgView.image=[self createNonInterpolatedUIImageFormCIImage:outputImage withSize:100.0];



//如果还想加上阴影,就在ImageView的Layer上使用下面代码添加阴影

_imgView.layer.shadowOffset=CGSizeMake(0, 0.5);//设置阴影的偏移量

_imgView.layer.shadowRadius=1;//设置阴影的半径

_imgView.layer.shadowColor=[UIColor blackColor].CGColor;//设置阴影的颜色为黑色

_imgView.layer.shadowOpacity=0.3;

}

//改变二维码大小

  • (UIImage )createNonInterpolatedUIImageFormCIImage:(CIImage )image withSize:(CGFloat) size {

    CGRect extent = CGRectIntegral(image.extent);

    CGFloat scale = MIN(size/CGRectGetWidth(extent), size/CGRectGetHeight(extent));

    // 创建bitmap;

    size_t width = CGRectGetWidth(extent) * scale;

    size_t height = CGRectGetHeight(extent) * scale;

    CGColorSpaceRef cs = CGColorSpaceCreateDeviceGray();

    CGContextRef bitmapRef = CGBitmapContextCreate(nil, width, height, 8, 0, cs, (CGBitmapInfo)kCGImageAlphaNone);

    CIContext *context = [CIContext contextWithOptions:nil];

    CGImageRef bitmapImage = [context createCGImage:image fromRect:extent];

    CGContextSetInterpolationQuality(bitmapRef, kCGInterpolationNone);

    CGContextScaleCTM(bitmapRef, scale, scale);

    CGContextDrawImage(bitmapRef, extent, bitmapImage);

    // 保存bitmap到图片

    CGImageRef scaledImage = CGBitmapContextCreateImage(bitmapRef);

    CGContextRelease(bitmapRef);

    CGImageRelease(bitmapImage);

    return [UIImage imageWithCGImage:scaledImage];

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值